xref: /openbmc/bmcweb/redfish-core/lib/chassis.hpp (revision 102a4cda)
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 "app.hpp"
19 #include "dbus_utility.hpp"
20 #include "led.hpp"
21 #include "query.hpp"
22 #include "redfish_util.hpp"
23 #include "registries/privilege_registry.hpp"
24 #include "utils/collection.hpp"
25 #include "utils/dbus_utils.hpp"
26 #include "utils/json_utils.hpp"
27 
28 #include <boost/system/error_code.hpp>
29 #include <boost/url/format.hpp>
30 #include <sdbusplus/asio/property.hpp>
31 #include <sdbusplus/message.hpp>
32 #include <sdbusplus/unpack_properties.hpp>
33 
34 #include <array>
35 #include <ranges>
36 #include <string_view>
37 
38 namespace redfish
39 {
40 
41 /**
42  * @brief Retrieves resources over dbus to link to the chassis
43  *
44  * @param[in] asyncResp  - Shared pointer for completing asynchronous
45  * calls
46  * @param[in] path       - Chassis dbus path to look for the storage.
47  *
48  * Calls the Association endpoints on the path + "/storage" and add the link of
49  * json["Links"]["Storage@odata.count"] =
50  *    {"@odata.id", "/redfish/v1/Storage/" + resourceId}
51  *
52  * @return None.
53  */
54 inline void getStorageLink(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
55                            const sdbusplus::message::object_path& path)
56 {
57     sdbusplus::asio::getProperty<std::vector<std::string>>(
58         *crow::connections::systemBus, "xyz.openbmc_project.ObjectMapper",
59         (path / "storage").str, "xyz.openbmc_project.Association", "endpoints",
60         [asyncResp](const boost::system::error_code& ec,
61                     const std::vector<std::string>& storageList) {
62         if (ec)
63         {
64             BMCWEB_LOG_DEBUG("getStorageLink got DBUS response error");
65             return;
66         }
67 
68         nlohmann::json::array_t storages;
69         for (const std::string& storagePath : storageList)
70         {
71             std::string id =
72                 sdbusplus::message::object_path(storagePath).filename();
73             if (id.empty())
74             {
75                 continue;
76             }
77 
78             nlohmann::json::object_t storage;
79             storage["@odata.id"] = boost::urls::format(
80                 "/redfish/v1/Systems/system/Storage/{}", id);
81             storages.emplace_back(std::move(storage));
82         }
83         asyncResp->res.jsonValue["Links"]["Storage@odata.count"] =
84             storages.size();
85         asyncResp->res.jsonValue["Links"]["Storage"] = std::move(storages);
86     });
87 }
88 
89 /**
90  * @brief Retrieves chassis state properties over dbus
91  *
92  * @param[in] asyncResp - Shared pointer for completing asynchronous calls.
93  *
94  * @return None.
95  */
96 inline void getChassisState(std::shared_ptr<bmcweb::AsyncResp> asyncResp)
97 {
98     // crow::connections::systemBus->async_method_call(
99     sdbusplus::asio::getProperty<std::string>(
100         *crow::connections::systemBus, "xyz.openbmc_project.State.Chassis",
101         "/xyz/openbmc_project/state/chassis0",
102         "xyz.openbmc_project.State.Chassis", "CurrentPowerState",
103         [asyncResp{std::move(asyncResp)}](const boost::system::error_code& ec,
104                                           const std::string& chassisState) {
105         if (ec)
106         {
107             if (ec == boost::system::errc::host_unreachable)
108             {
109                 // Service not available, no error, just don't return
110                 // chassis state info
111                 BMCWEB_LOG_DEBUG("Service not available {}", ec);
112                 return;
113             }
114             BMCWEB_LOG_DEBUG("DBUS response error {}", ec);
115             messages::internalError(asyncResp->res);
116             return;
117         }
118 
119         BMCWEB_LOG_DEBUG("Chassis state: {}", chassisState);
120         // Verify Chassis State
121         if (chassisState == "xyz.openbmc_project.State.Chassis.PowerState.On")
122         {
123             asyncResp->res.jsonValue["PowerState"] = "On";
124             asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
125         }
126         else if (chassisState ==
127                  "xyz.openbmc_project.State.Chassis.PowerState.Off")
128         {
129             asyncResp->res.jsonValue["PowerState"] = "Off";
130             asyncResp->res.jsonValue["Status"]["State"] = "StandbyOffline";
131         }
132     });
133 }
134 
135 /**
136  * Retrieves physical security properties over dbus
137  */
138 inline void handlePhysicalSecurityGetSubTree(
139     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
140     const boost::system::error_code& ec,
141     const dbus::utility::MapperGetSubTreeResponse& subtree)
142 {
143     if (ec)
144     {
145         // do not add err msg in redfish response, because this is not
146         //     mandatory property
147         BMCWEB_LOG_INFO("DBUS error: no matched iface {}", ec);
148         return;
149     }
150     // Iterate over all retrieved ObjectPaths.
151     for (const auto& object : subtree)
152     {
153         if (!object.second.empty())
154         {
155             const auto& service = object.second.front();
156 
157             BMCWEB_LOG_DEBUG("Get intrusion status by service ");
158 
159             sdbusplus::asio::getProperty<std::string>(
160                 *crow::connections::systemBus, service.first, object.first,
161                 "xyz.openbmc_project.Chassis.Intrusion", "Status",
162                 [asyncResp](const boost::system::error_code& ec1,
163                             const std::string& value) {
164                 if (ec1)
165                 {
166                     // do not add err msg in redfish response, because this is
167                     // not
168                     //     mandatory property
169                     BMCWEB_LOG_ERROR("DBUS response error {}", ec1);
170                     return;
171                 }
172                 asyncResp->res
173                     .jsonValue["PhysicalSecurity"]["IntrusionSensorNumber"] = 1;
174                 asyncResp->res
175                     .jsonValue["PhysicalSecurity"]["IntrusionSensor"] = value;
176             });
177 
178             return;
179         }
180     }
181 }
182 
183 inline void handleChassisCollectionGet(
184     App& app, const crow::Request& req,
185     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
186 {
187     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
188     {
189         return;
190     }
191     asyncResp->res.jsonValue["@odata.type"] =
192         "#ChassisCollection.ChassisCollection";
193     asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Chassis";
194     asyncResp->res.jsonValue["Name"] = "Chassis Collection";
195 
196     constexpr std::array<std::string_view, 2> interfaces{
197         "xyz.openbmc_project.Inventory.Item.Board",
198         "xyz.openbmc_project.Inventory.Item.Chassis"};
199     collection_util::getCollectionMembers(
200         asyncResp, boost::urls::url("/redfish/v1/Chassis"), interfaces,
201         "/xyz/openbmc_project/inventory");
202 }
203 
204 inline void getChassisContainedBy(
205     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
206     const std::string& chassisId, const boost::system::error_code& ec,
207     const dbus::utility::MapperEndPoints& upstreamChassisPaths)
208 {
209     if (ec)
210     {
211         if (ec.value() != EBADR)
212         {
213             BMCWEB_LOG_ERROR("DBUS response error {}", ec);
214             messages::internalError(asyncResp->res);
215         }
216         return;
217     }
218     if (upstreamChassisPaths.empty())
219     {
220         return;
221     }
222     if (upstreamChassisPaths.size() > 1)
223     {
224         BMCWEB_LOG_ERROR("{} is contained by multiple chassis", chassisId);
225         messages::internalError(asyncResp->res);
226         return;
227     }
228 
229     sdbusplus::message::object_path upstreamChassisPath(
230         upstreamChassisPaths[0]);
231     std::string upstreamChassis = upstreamChassisPath.filename();
232     if (upstreamChassis.empty())
233     {
234         BMCWEB_LOG_WARNING("Malformed upstream Chassis path {} on {}",
235                            upstreamChassisPath.str, chassisId);
236         return;
237     }
238 
239     asyncResp->res.jsonValue["Links"]["ContainedBy"]["@odata.id"] =
240         boost::urls::format("/redfish/v1/Chassis/{}", upstreamChassis);
241 }
242 
243 inline void getChassisContains(
244     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
245     const std::string& chassisId, const boost::system::error_code& ec,
246     const dbus::utility::MapperEndPoints& downstreamChassisPaths)
247 {
248     if (ec)
249     {
250         if (ec.value() != EBADR)
251         {
252             BMCWEB_LOG_ERROR("DBUS response error {}", ec);
253             messages::internalError(asyncResp->res);
254         }
255         return;
256     }
257     if (downstreamChassisPaths.empty())
258     {
259         return;
260     }
261     nlohmann::json& jValue = asyncResp->res.jsonValue["Links"]["Contains"];
262     if (!jValue.is_array())
263     {
264         // Create the array if it was empty
265         jValue = nlohmann::json::array();
266     }
267     for (const auto& p : downstreamChassisPaths)
268     {
269         sdbusplus::message::object_path downstreamChassisPath(p);
270         std::string downstreamChassis = downstreamChassisPath.filename();
271         if (downstreamChassis.empty())
272         {
273             BMCWEB_LOG_WARNING("Malformed downstream Chassis path {} on {}",
274                                downstreamChassisPath.str, chassisId);
275             continue;
276         }
277         nlohmann::json link;
278         link["@odata.id"] = boost::urls::format("/redfish/v1/Chassis/{}",
279                                                 downstreamChassis);
280         jValue.push_back(std::move(link));
281     }
282     asyncResp->res.jsonValue["Links"]["Contains@odata.count"] = jValue.size();
283 }
284 
285 inline void
286     getChassisConnectivity(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
287                            const std::string& chassisId,
288                            const std::string& chassisPath)
289 {
290     BMCWEB_LOG_DEBUG("Get chassis connectivity");
291 
292     dbus::utility::getAssociationEndPoints(
293         chassisPath + "/contained_by",
294         std::bind_front(getChassisContainedBy, asyncResp, chassisId));
295 
296     dbus::utility::getAssociationEndPoints(
297         chassisPath + "/containing",
298         std::bind_front(getChassisContains, asyncResp, chassisId));
299 }
300 
301 /**
302  * ChassisCollection derived class for delivering Chassis Collection Schema
303  *  Functions triggers appropriate requests on DBus
304  */
305 inline void requestRoutesChassisCollection(App& app)
306 {
307     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/")
308         .privileges(redfish::privileges::getChassisCollection)
309         .methods(boost::beast::http::verb::get)(
310             std::bind_front(handleChassisCollectionGet, std::ref(app)));
311 }
312 
313 inline void
314     getChassisLocationCode(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
315                            const std::string& connectionName,
316                            const std::string& path)
317 {
318     sdbusplus::asio::getProperty<std::string>(
319         *crow::connections::systemBus, connectionName, path,
320         "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode",
321         [asyncResp](const boost::system::error_code& ec,
322                     const std::string& property) {
323         if (ec)
324         {
325             BMCWEB_LOG_ERROR("DBUS response error for Location");
326             messages::internalError(asyncResp->res);
327             return;
328         }
329 
330         asyncResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] =
331             property;
332     });
333 }
334 
335 inline void getChassisUUID(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
336                            const std::string& connectionName,
337                            const std::string& path)
338 {
339     sdbusplus::asio::getProperty<std::string>(
340         *crow::connections::systemBus, connectionName, path,
341         "xyz.openbmc_project.Common.UUID", "UUID",
342         [asyncResp](const boost::system::error_code& ec,
343                     const std::string& chassisUUID) {
344         if (ec)
345         {
346             BMCWEB_LOG_ERROR("DBUS response error for UUID");
347             messages::internalError(asyncResp->res);
348             return;
349         }
350         asyncResp->res.jsonValue["UUID"] = chassisUUID;
351     });
352 }
353 
354 inline void handleDecoratorAssetProperties(
355     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
356     const std::string& chassisId, const std::string& path,
357     const dbus::utility::DBusPropertiesMap& propertiesList)
358 {
359     const std::string* partNumber = nullptr;
360     const std::string* serialNumber = nullptr;
361     const std::string* manufacturer = nullptr;
362     const std::string* model = nullptr;
363     const std::string* sparePartNumber = nullptr;
364 
365     const bool success = sdbusplus::unpackPropertiesNoThrow(
366         dbus_utils::UnpackErrorPrinter(), propertiesList, "PartNumber",
367         partNumber, "SerialNumber", serialNumber, "Manufacturer", manufacturer,
368         "Model", model, "SparePartNumber", sparePartNumber);
369 
370     if (!success)
371     {
372         messages::internalError(asyncResp->res);
373         return;
374     }
375 
376     if (partNumber != nullptr)
377     {
378         asyncResp->res.jsonValue["PartNumber"] = *partNumber;
379     }
380 
381     if (serialNumber != nullptr)
382     {
383         asyncResp->res.jsonValue["SerialNumber"] = *serialNumber;
384     }
385 
386     if (manufacturer != nullptr)
387     {
388         asyncResp->res.jsonValue["Manufacturer"] = *manufacturer;
389     }
390 
391     if (model != nullptr)
392     {
393         asyncResp->res.jsonValue["Model"] = *model;
394     }
395 
396     // SparePartNumber is optional on D-Bus
397     // so skip if it is empty
398     if (sparePartNumber != nullptr && !sparePartNumber->empty())
399     {
400         asyncResp->res.jsonValue["SparePartNumber"] = *sparePartNumber;
401     }
402 
403     asyncResp->res.jsonValue["Name"] = chassisId;
404     asyncResp->res.jsonValue["Id"] = chassisId;
405 
406     if constexpr (BMCWEB_REDFISH_ALLOW_DEPRECATED_POWER_THERMAL)
407     {
408         asyncResp->res.jsonValue["Thermal"]["@odata.id"] =
409             boost::urls::format("/redfish/v1/Chassis/{}/Thermal", chassisId);
410         // Power object
411         asyncResp->res.jsonValue["Power"]["@odata.id"] =
412             boost::urls::format("/redfish/v1/Chassis/{}/Power", chassisId);
413     }
414 
415     if constexpr (BMCWEB_REDFISH_NEW_POWERSUBSYSTEM_THERMALSUBSYSTEM)
416     {
417         asyncResp->res.jsonValue["ThermalSubsystem"]["@odata.id"] =
418             boost::urls::format("/redfish/v1/Chassis/{}/ThermalSubsystem",
419                                 chassisId);
420         asyncResp->res.jsonValue["PowerSubsystem"]["@odata.id"] =
421             boost::urls::format("/redfish/v1/Chassis/{}/PowerSubsystem",
422                                 chassisId);
423         asyncResp->res.jsonValue["EnvironmentMetrics"]["@odata.id"] =
424             boost::urls::format("/redfish/v1/Chassis/{}/EnvironmentMetrics",
425                                 chassisId);
426     }
427     // SensorCollection
428     asyncResp->res.jsonValue["Sensors"]["@odata.id"] =
429         boost::urls::format("/redfish/v1/Chassis/{}/Sensors", chassisId);
430     asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
431 
432     nlohmann::json::array_t computerSystems;
433     nlohmann::json::object_t system;
434     system["@odata.id"] = "/redfish/v1/Systems/system";
435     computerSystems.emplace_back(std::move(system));
436     asyncResp->res.jsonValue["Links"]["ComputerSystems"] =
437         std::move(computerSystems);
438 
439     nlohmann::json::array_t managedBy;
440     nlohmann::json::object_t manager;
441     manager["@odata.id"] = "/redfish/v1/Managers/bmc";
442     managedBy.emplace_back(std::move(manager));
443     asyncResp->res.jsonValue["Links"]["ManagedBy"] = std::move(managedBy);
444     getChassisState(asyncResp);
445     getStorageLink(asyncResp, path);
446 }
447 
448 inline void handleChassisGetSubTree(
449     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
450     const std::string& chassisId, const boost::system::error_code& ec,
451     const dbus::utility::MapperGetSubTreeResponse& subtree)
452 {
453     if (ec)
454     {
455         BMCWEB_LOG_ERROR("DBUS response error {}", ec);
456         messages::internalError(asyncResp->res);
457         return;
458     }
459     // Iterate over all retrieved ObjectPaths.
460     for (const std::pair<
461              std::string,
462              std::vector<std::pair<std::string, std::vector<std::string>>>>&
463              object : subtree)
464     {
465         const std::string& path = object.first;
466         const std::vector<std::pair<std::string, std::vector<std::string>>>&
467             connectionNames = object.second;
468 
469         sdbusplus::message::object_path objPath(path);
470         if (objPath.filename() != chassisId)
471         {
472             continue;
473         }
474 
475         getChassisConnectivity(asyncResp, chassisId, path);
476 
477         if (connectionNames.empty())
478         {
479             BMCWEB_LOG_ERROR("Got 0 Connection names");
480             continue;
481         }
482 
483         asyncResp->res.jsonValue["@odata.type"] = "#Chassis.v1_22_0.Chassis";
484         asyncResp->res.jsonValue["@odata.id"] =
485             boost::urls::format("/redfish/v1/Chassis/{}", chassisId);
486         asyncResp->res.jsonValue["Name"] = "Chassis Collection";
487         asyncResp->res.jsonValue["ChassisType"] = "RackMount";
488         asyncResp->res.jsonValue["Actions"]["#Chassis.Reset"]["target"] =
489             boost::urls::format("/redfish/v1/Chassis/{}/Actions/Chassis.Reset",
490                                 chassisId);
491         asyncResp->res
492             .jsonValue["Actions"]["#Chassis.Reset"]["@Redfish.ActionInfo"] =
493             boost::urls::format("/redfish/v1/Chassis/{}/ResetActionInfo",
494                                 chassisId);
495         dbus::utility::getAssociationEndPoints(
496             path + "/drive",
497             [asyncResp, chassisId](const boost::system::error_code& ec3,
498                                    const dbus::utility::MapperEndPoints& resp) {
499             if (ec3 || resp.empty())
500             {
501                 return; // no drives = no failures
502             }
503 
504             nlohmann::json reference;
505             reference["@odata.id"] =
506                 boost::urls::format("/redfish/v1/Chassis/{}/Drives", chassisId);
507             asyncResp->res.jsonValue["Drives"] = std::move(reference);
508         });
509 
510         const std::string& connectionName = connectionNames[0].first;
511 
512         const std::vector<std::string>& interfaces2 = connectionNames[0].second;
513         const std::array<const char*, 3> hasIndicatorLed = {
514             "xyz.openbmc_project.Inventory.Item.Chassis",
515             "xyz.openbmc_project.Inventory.Item.Panel",
516             "xyz.openbmc_project.Inventory.Item.Board.Motherboard"};
517 
518         const std::string assetTagInterface =
519             "xyz.openbmc_project.Inventory.Decorator.AssetTag";
520         const std::string replaceableInterface =
521             "xyz.openbmc_project.Inventory.Decorator.Replaceable";
522         const std::string revisionInterface =
523             "xyz.openbmc_project.Inventory.Decorator.Revision";
524         for (const auto& interface : interfaces2)
525         {
526             if (interface == assetTagInterface)
527             {
528                 sdbusplus::asio::getProperty<std::string>(
529                     *crow::connections::systemBus, connectionName, path,
530                     assetTagInterface, "AssetTag",
531                     [asyncResp, chassisId](const boost::system::error_code& ec2,
532                                            const std::string& property) {
533                     if (ec2)
534                     {
535                         BMCWEB_LOG_ERROR("DBus response error for AssetTag: {}",
536                                          ec2);
537                         messages::internalError(asyncResp->res);
538                         return;
539                     }
540                     asyncResp->res.jsonValue["AssetTag"] = property;
541                 });
542             }
543             else if (interface == replaceableInterface)
544             {
545                 sdbusplus::asio::getProperty<bool>(
546                     *crow::connections::systemBus, connectionName, path,
547                     replaceableInterface, "HotPluggable",
548                     [asyncResp, chassisId](const boost::system::error_code& ec2,
549                                            const bool property) {
550                     if (ec2)
551                     {
552                         BMCWEB_LOG_ERROR(
553                             "DBus response error for HotPluggable: {}", ec2);
554                         messages::internalError(asyncResp->res);
555                         return;
556                     }
557                     asyncResp->res.jsonValue["HotPluggable"] = property;
558                 });
559             }
560             else if (interface == revisionInterface)
561             {
562                 sdbusplus::asio::getProperty<std::string>(
563                     *crow::connections::systemBus, connectionName, path,
564                     revisionInterface, "Version",
565                     [asyncResp, chassisId](const boost::system::error_code& ec2,
566                                            const std::string& property) {
567                     if (ec2)
568                     {
569                         BMCWEB_LOG_ERROR("DBus response error for Version: {}",
570                                          ec2);
571                         messages::internalError(asyncResp->res);
572                         return;
573                     }
574                     asyncResp->res.jsonValue["Version"] = property;
575                 });
576             }
577         }
578 
579         for (const char* interface : hasIndicatorLed)
580         {
581             if (std::ranges::find(interfaces2, interface) != interfaces2.end())
582             {
583                 getIndicatorLedState(asyncResp);
584                 getSystemLocationIndicatorActive(asyncResp);
585                 break;
586             }
587         }
588 
589         sdbusplus::asio::getAllProperties(
590             *crow::connections::systemBus, connectionName, path,
591             "xyz.openbmc_project.Inventory.Decorator.Asset",
592             [asyncResp, chassisId,
593              path](const boost::system::error_code&,
594                    const dbus::utility::DBusPropertiesMap& propertiesList) {
595             handleDecoratorAssetProperties(asyncResp, chassisId, path,
596                                            propertiesList);
597         });
598 
599         for (const auto& interface : interfaces2)
600         {
601             if (interface == "xyz.openbmc_project.Common.UUID")
602             {
603                 getChassisUUID(asyncResp, connectionName, path);
604             }
605             else if (interface ==
606                      "xyz.openbmc_project.Inventory.Decorator.LocationCode")
607             {
608                 getChassisLocationCode(asyncResp, connectionName, path);
609             }
610         }
611 
612         return;
613     }
614 
615     // Couldn't find an object with that name.  return an error
616     messages::resourceNotFound(asyncResp->res, "Chassis", chassisId);
617 }
618 
619 inline void
620     handleChassisGet(App& app, const crow::Request& req,
621                      const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
622                      const std::string& chassisId)
623 {
624     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
625     {
626         return;
627     }
628     constexpr std::array<std::string_view, 2> interfaces = {
629         "xyz.openbmc_project.Inventory.Item.Board",
630         "xyz.openbmc_project.Inventory.Item.Chassis"};
631 
632     dbus::utility::getSubTree(
633         "/xyz/openbmc_project/inventory", 0, interfaces,
634         std::bind_front(handleChassisGetSubTree, asyncResp, chassisId));
635 
636     constexpr std::array<std::string_view, 1> interfaces2 = {
637         "xyz.openbmc_project.Chassis.Intrusion"};
638 
639     dbus::utility::getSubTree(
640         "/xyz/openbmc_project", 0, interfaces2,
641         std::bind_front(handlePhysicalSecurityGetSubTree, asyncResp));
642 }
643 
644 inline void
645     handleChassisPatch(App& app, const crow::Request& req,
646                        const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
647                        const std::string& param)
648 {
649     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
650     {
651         return;
652     }
653     std::optional<bool> locationIndicatorActive;
654     std::optional<std::string> indicatorLed;
655 
656     if (param.empty())
657     {
658         return;
659     }
660 
661     if (!json_util::readJsonPatch(
662             req, asyncResp->res, "LocationIndicatorActive",
663             locationIndicatorActive, "IndicatorLED", indicatorLed))
664     {
665         return;
666     }
667 
668     // TODO (Gunnar): Remove IndicatorLED after enough time has passed
669     if (!locationIndicatorActive && !indicatorLed)
670     {
671         return; // delete this when we support more patch properties
672     }
673     if (indicatorLed)
674     {
675         asyncResp->res.addHeader(
676             boost::beast::http::field::warning,
677             "299 - \"IndicatorLED is deprecated. Use LocationIndicatorActive instead.\"");
678     }
679 
680     constexpr std::array<std::string_view, 2> interfaces = {
681         "xyz.openbmc_project.Inventory.Item.Board",
682         "xyz.openbmc_project.Inventory.Item.Chassis"};
683 
684     const std::string& chassisId = param;
685 
686     dbus::utility::getSubTree(
687         "/xyz/openbmc_project/inventory", 0, interfaces,
688         [asyncResp, chassisId, locationIndicatorActive,
689          indicatorLed](const boost::system::error_code& ec,
690                        const dbus::utility::MapperGetSubTreeResponse& subtree) {
691         if (ec)
692         {
693             BMCWEB_LOG_ERROR("DBUS response error {}", ec);
694             messages::internalError(asyncResp->res);
695             return;
696         }
697 
698         // Iterate over all retrieved ObjectPaths.
699         for (const std::pair<
700                  std::string,
701                  std::vector<std::pair<std::string, std::vector<std::string>>>>&
702                  object : subtree)
703         {
704             const std::string& path = object.first;
705             const std::vector<std::pair<std::string, std::vector<std::string>>>&
706                 connectionNames = object.second;
707 
708             sdbusplus::message::object_path objPath(path);
709             if (objPath.filename() != chassisId)
710             {
711                 continue;
712             }
713 
714             if (connectionNames.empty())
715             {
716                 BMCWEB_LOG_ERROR("Got 0 Connection names");
717                 continue;
718             }
719 
720             const std::vector<std::string>& interfaces3 =
721                 connectionNames[0].second;
722 
723             const std::array<const char*, 3> hasIndicatorLed = {
724                 "xyz.openbmc_project.Inventory.Item.Chassis",
725                 "xyz.openbmc_project.Inventory.Item.Panel",
726                 "xyz.openbmc_project.Inventory.Item.Board.Motherboard"};
727             bool indicatorChassis = false;
728             for (const char* interface : hasIndicatorLed)
729             {
730                 if (std::ranges::find(interfaces3, interface) !=
731                     interfaces3.end())
732                 {
733                     indicatorChassis = true;
734                     break;
735                 }
736             }
737             if (locationIndicatorActive)
738             {
739                 if (indicatorChassis)
740                 {
741                     setSystemLocationIndicatorActive(asyncResp,
742                                                      *locationIndicatorActive);
743                 }
744                 else
745                 {
746                     messages::propertyUnknown(asyncResp->res,
747                                               "LocationIndicatorActive");
748                 }
749             }
750             if (indicatorLed)
751             {
752                 if (indicatorChassis)
753                 {
754                     setIndicatorLedState(asyncResp, *indicatorLed);
755                 }
756                 else
757                 {
758                     messages::propertyUnknown(asyncResp->res, "IndicatorLED");
759                 }
760             }
761             return;
762         }
763 
764         messages::resourceNotFound(asyncResp->res, "Chassis", chassisId);
765     });
766 }
767 
768 /**
769  * Chassis override class for delivering Chassis Schema
770  * Functions triggers appropriate requests on DBus
771  */
772 inline void requestRoutesChassis(App& app)
773 {
774     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/")
775         .privileges(redfish::privileges::getChassis)
776         .methods(boost::beast::http::verb::get)(
777             std::bind_front(handleChassisGet, std::ref(app)));
778 
779     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/")
780         .privileges(redfish::privileges::patchChassis)
781         .methods(boost::beast::http::verb::patch)(
782             std::bind_front(handleChassisPatch, std::ref(app)));
783 }
784 
785 inline void
786     doChassisPowerCycle(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
787 {
788     constexpr std::array<std::string_view, 1> interfaces = {
789         "xyz.openbmc_project.State.Chassis"};
790 
791     // Use mapper to get subtree paths.
792     dbus::utility::getSubTreePaths(
793         "/", 0, interfaces,
794         [asyncResp](
795             const boost::system::error_code& ec,
796             const dbus::utility::MapperGetSubTreePathsResponse& chassisList) {
797         if (ec)
798         {
799             BMCWEB_LOG_ERROR("[mapper] Bad D-Bus request error: {}", ec);
800             messages::internalError(asyncResp->res);
801             return;
802         }
803 
804         const char* processName = "xyz.openbmc_project.State.Chassis";
805         const char* interfaceName = "xyz.openbmc_project.State.Chassis";
806         const char* destProperty = "RequestedPowerTransition";
807         const std::string propertyValue =
808             "xyz.openbmc_project.State.Chassis.Transition.PowerCycle";
809         std::string objectPath = "/xyz/openbmc_project/state/chassis_system0";
810 
811         /* Look for system reset chassis path */
812         if ((std::ranges::find(chassisList, objectPath)) == chassisList.end())
813         {
814             /* We prefer to reset the full chassis_system, but if it doesn't
815              * exist on some platforms, fall back to a host-only power reset
816              */
817             objectPath = "/xyz/openbmc_project/state/chassis0";
818         }
819 
820         setDbusProperty(asyncResp, processName, objectPath, interfaceName,
821                         destProperty, "ResetType", propertyValue);
822     });
823 }
824 
825 inline void handleChassisResetActionInfoPost(
826     App& app, const crow::Request& req,
827     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
828     const std::string& /*chassisId*/)
829 {
830     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
831     {
832         return;
833     }
834     BMCWEB_LOG_DEBUG("Post Chassis Reset.");
835 
836     std::string resetType;
837 
838     if (!json_util::readJsonAction(req, asyncResp->res, "ResetType", resetType))
839     {
840         return;
841     }
842 
843     if (resetType != "PowerCycle")
844     {
845         BMCWEB_LOG_DEBUG("Invalid property value for ResetType: {}", resetType);
846         messages::actionParameterNotSupported(asyncResp->res, resetType,
847                                               "ResetType");
848 
849         return;
850     }
851     doChassisPowerCycle(asyncResp);
852 }
853 
854 /**
855  * ChassisResetAction class supports the POST method for the Reset
856  * action.
857  * Function handles POST method request.
858  * Analyzes POST body before sending Reset request data to D-Bus.
859  */
860 
861 inline void requestRoutesChassisResetAction(App& app)
862 {
863     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Actions/Chassis.Reset/")
864         .privileges(redfish::privileges::postChassis)
865         .methods(boost::beast::http::verb::post)(
866             std::bind_front(handleChassisResetActionInfoPost, std::ref(app)));
867 }
868 
869 inline void handleChassisResetActionInfoGet(
870     App& app, const crow::Request& req,
871     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
872     const std::string& chassisId)
873 {
874     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
875     {
876         return;
877     }
878     asyncResp->res.jsonValue["@odata.type"] = "#ActionInfo.v1_1_2.ActionInfo";
879     asyncResp->res.jsonValue["@odata.id"] = boost::urls::format(
880         "/redfish/v1/Chassis/{}/ResetActionInfo", chassisId);
881     asyncResp->res.jsonValue["Name"] = "Reset Action Info";
882 
883     asyncResp->res.jsonValue["Id"] = "ResetActionInfo";
884     nlohmann::json::array_t parameters;
885     nlohmann::json::object_t parameter;
886     parameter["Name"] = "ResetType";
887     parameter["Required"] = true;
888     parameter["DataType"] = "String";
889     nlohmann::json::array_t allowed;
890     allowed.emplace_back("PowerCycle");
891     parameter["AllowableValues"] = std::move(allowed);
892     parameters.emplace_back(std::move(parameter));
893 
894     asyncResp->res.jsonValue["Parameters"] = std::move(parameters);
895 }
896 
897 /**
898  * ChassisResetActionInfo derived class for delivering Chassis
899  * ResetType AllowableValues using ResetInfo schema.
900  */
901 inline void requestRoutesChassisResetActionInfo(App& app)
902 {
903     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/ResetActionInfo/")
904         .privileges(redfish::privileges::getActionInfo)
905         .methods(boost::beast::http::verb::get)(
906             std::bind_front(handleChassisResetActionInfoGet, std::ref(app)));
907 }
908 
909 } // namespace redfish
910