xref: /openbmc/bmcweb/redfish-core/lib/sensors.hpp (revision 81ee0e74bd3a575593ee2a21de560936843f526a)
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_singleton.hpp"
20 #include "dbus_utility.hpp"
21 #include "generated/enums/redundancy.hpp"
22 #include "generated/enums/resource.hpp"
23 #include "query.hpp"
24 #include "registries/privilege_registry.hpp"
25 #include "str_utility.hpp"
26 #include "utils/dbus_utils.hpp"
27 #include "utils/json_utils.hpp"
28 #include "utils/query_param.hpp"
29 #include "utils/sensor_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/unpack_properties.hpp>
35 
36 #include <array>
37 #include <cmath>
38 #include <iterator>
39 #include <limits>
40 #include <map>
41 #include <ranges>
42 #include <set>
43 #include <string>
44 #include <string_view>
45 #include <utility>
46 #include <variant>
47 
48 namespace redfish
49 {
50 
51 namespace sensors
52 {
53 
54 // clang-format off
55 namespace dbus
56 {
57 constexpr auto powerPaths = std::to_array<std::string_view>({
58     "/xyz/openbmc_project/sensors/voltage",
59     "/xyz/openbmc_project/sensors/power"
60 });
61 
62 constexpr auto getSensorPaths(){
63     if constexpr(BMCWEB_REDFISH_NEW_POWERSUBSYSTEM_THERMALSUBSYSTEM){
64     return std::to_array<std::string_view>({
65         "/xyz/openbmc_project/sensors/power",
66         "/xyz/openbmc_project/sensors/current",
67         "/xyz/openbmc_project/sensors/airflow",
68         "/xyz/openbmc_project/sensors/humidity",
69         "/xyz/openbmc_project/sensors/voltage",
70         "/xyz/openbmc_project/sensors/fan_tach",
71         "/xyz/openbmc_project/sensors/temperature",
72         "/xyz/openbmc_project/sensors/fan_pwm",
73         "/xyz/openbmc_project/sensors/altitude",
74         "/xyz/openbmc_project/sensors/energy",
75         "/xyz/openbmc_project/sensors/utilization"});
76     } else {
77       return  std::to_array<std::string_view>({"/xyz/openbmc_project/sensors/power",
78         "/xyz/openbmc_project/sensors/current",
79         "/xyz/openbmc_project/sensors/airflow",
80         "/xyz/openbmc_project/sensors/humidity",
81         "/xyz/openbmc_project/sensors/utilization"});
82 }
83 }
84 
85 constexpr auto sensorPaths = getSensorPaths();
86 
87 constexpr auto thermalPaths = std::to_array<std::string_view>({
88     "/xyz/openbmc_project/sensors/fan_tach",
89     "/xyz/openbmc_project/sensors/temperature",
90     "/xyz/openbmc_project/sensors/fan_pwm"
91 });
92 
93 } // namespace dbus
94 // clang-format on
95 
96 constexpr std::string_view powerNodeStr = sensor_utils::chassisSubNodeToString(
97     sensor_utils::ChassisSubNode::powerNode);
98 constexpr std::string_view sensorsNodeStr =
99     sensor_utils::chassisSubNodeToString(
100         sensor_utils::ChassisSubNode::sensorsNode);
101 constexpr std::string_view thermalNodeStr =
102     sensor_utils::chassisSubNodeToString(
103         sensor_utils::ChassisSubNode::thermalNode);
104 
105 using sensorPair =
106     std::pair<std::string_view, std::span<const std::string_view>>;
107 static constexpr std::array<sensorPair, 3> paths = {
108     {{sensors::powerNodeStr, dbus::powerPaths},
109      {sensors::sensorsNodeStr, dbus::sensorPaths},
110      {sensors::thermalNodeStr, dbus::thermalPaths}}};
111 
112 } // namespace sensors
113 
114 /**
115  * SensorsAsyncResp
116  * Gathers data needed for response processing after async calls are done
117  */
118 class SensorsAsyncResp
119 {
120   public:
121     using DataCompleteCb = std::function<void(
122         const boost::beast::http::status status,
123         const std::map<std::string, std::string>& uriToDbus)>;
124 
125     struct SensorData
126     {
127         std::string name;
128         std::string uri;
129         std::string dbusPath;
130     };
131 
132     SensorsAsyncResp(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn,
133                      const std::string& chassisIdIn,
134                      std::span<const std::string_view> typesIn,
135                      std::string_view subNode) :
136         asyncResp(asyncRespIn), chassisId(chassisIdIn), types(typesIn),
137         chassisSubNode(subNode), efficientExpand(false)
138     {}
139 
140     // Store extra data about sensor mapping and return it in callback
141     SensorsAsyncResp(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn,
142                      const std::string& chassisIdIn,
143                      std::span<const std::string_view> typesIn,
144                      std::string_view subNode,
145                      DataCompleteCb&& creationComplete) :
146         asyncResp(asyncRespIn), chassisId(chassisIdIn), types(typesIn),
147         chassisSubNode(subNode), efficientExpand(false),
148         metadata{std::vector<SensorData>()},
149         dataComplete{std::move(creationComplete)}
150     {}
151 
152     // sensor collections expand
153     SensorsAsyncResp(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn,
154                      const std::string& chassisIdIn,
155                      std::span<const std::string_view> typesIn,
156                      const std::string_view& subNode, bool efficientExpandIn) :
157         asyncResp(asyncRespIn), chassisId(chassisIdIn), types(typesIn),
158         chassisSubNode(subNode), efficientExpand(efficientExpandIn)
159     {}
160 
161     ~SensorsAsyncResp()
162     {
163         if (asyncResp->res.result() ==
164             boost::beast::http::status::internal_server_error)
165         {
166             // Reset the json object to clear out any data that made it in
167             // before the error happened todo(ed) handle error condition with
168             // proper code
169             asyncResp->res.jsonValue = nlohmann::json::object();
170         }
171 
172         if (dataComplete && metadata)
173         {
174             std::map<std::string, std::string> map;
175             if (asyncResp->res.result() == boost::beast::http::status::ok)
176             {
177                 for (auto& sensor : *metadata)
178                 {
179                     map.emplace(sensor.uri, sensor.dbusPath);
180                 }
181             }
182             dataComplete(asyncResp->res.result(), map);
183         }
184     }
185 
186     SensorsAsyncResp(const SensorsAsyncResp&) = delete;
187     SensorsAsyncResp(SensorsAsyncResp&&) = delete;
188     SensorsAsyncResp& operator=(const SensorsAsyncResp&) = delete;
189     SensorsAsyncResp& operator=(SensorsAsyncResp&&) = delete;
190 
191     void addMetadata(const nlohmann::json& sensorObject,
192                      const std::string& dbusPath)
193     {
194         if (metadata)
195         {
196             metadata->emplace_back(SensorData{
197                 sensorObject["Name"], sensorObject["@odata.id"], dbusPath});
198         }
199     }
200 
201     void updateUri(const std::string& name, const std::string& uri)
202     {
203         if (metadata)
204         {
205             for (auto& sensor : *metadata)
206             {
207                 if (sensor.name == name)
208                 {
209                     sensor.uri = uri;
210                 }
211             }
212         }
213     }
214 
215     const std::shared_ptr<bmcweb::AsyncResp> asyncResp;
216     const std::string chassisId;
217     const std::span<const std::string_view> types;
218     const std::string chassisSubNode;
219     const bool efficientExpand;
220 
221   private:
222     std::optional<std::vector<SensorData>> metadata;
223     DataCompleteCb dataComplete;
224 };
225 
226 using InventoryItem = sensor_utils::InventoryItem;
227 
228 /**
229  * @brief Get objects with connection necessary for sensors
230  * @param SensorsAsyncResp Pointer to object holding response data
231  * @param sensorNames Sensors retrieved from chassis
232  * @param callback Callback for processing gathered connections
233  */
234 template <typename Callback>
235 void getObjectsWithConnection(
236     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
237     const std::shared_ptr<std::set<std::string>>& sensorNames,
238     Callback&& callback)
239 {
240     BMCWEB_LOG_DEBUG("getObjectsWithConnection enter");
241     const std::string path = "/xyz/openbmc_project/sensors";
242     constexpr std::array<std::string_view, 1> interfaces = {
243         "xyz.openbmc_project.Sensor.Value"};
244 
245     // Make call to ObjectMapper to find all sensors objects
246     dbus::utility::getSubTree(
247         path, 2, interfaces,
248         [callback = std::forward<Callback>(callback), sensorsAsyncResp,
249          sensorNames](const boost::system::error_code& ec,
250                       const dbus::utility::MapperGetSubTreeResponse& subtree) {
251             // Response handler for parsing objects subtree
252             BMCWEB_LOG_DEBUG("getObjectsWithConnection resp_handler enter");
253             if (ec)
254             {
255                 messages::internalError(sensorsAsyncResp->asyncResp->res);
256                 BMCWEB_LOG_ERROR(
257                     "getObjectsWithConnection resp_handler: Dbus error {}", ec);
258                 return;
259             }
260 
261             BMCWEB_LOG_DEBUG("Found {} subtrees", subtree.size());
262 
263             // Make unique list of connections only for requested sensor types
264             // and found in the chassis
265             std::set<std::string> connections;
266             std::set<std::pair<std::string, std::string>> objectsWithConnection;
267 
268             BMCWEB_LOG_DEBUG("sensorNames list count: {}", sensorNames->size());
269             for (const std::string& tsensor : *sensorNames)
270             {
271                 BMCWEB_LOG_DEBUG("Sensor to find: {}", tsensor);
272             }
273 
274             for (const std::pair<std::string,
275                                  std::vector<std::pair<
276                                      std::string, std::vector<std::string>>>>&
277                      object : subtree)
278             {
279                 if (sensorNames->find(object.first) != sensorNames->end())
280                 {
281                     for (const std::pair<std::string, std::vector<std::string>>&
282                              objData : object.second)
283                     {
284                         BMCWEB_LOG_DEBUG("Adding connection: {}",
285                                          objData.first);
286                         connections.insert(objData.first);
287                         objectsWithConnection.insert(
288                             std::make_pair(object.first, objData.first));
289                     }
290                 }
291             }
292             BMCWEB_LOG_DEBUG("Found {} connections", connections.size());
293             callback(std::move(connections), std::move(objectsWithConnection));
294             BMCWEB_LOG_DEBUG("getObjectsWithConnection resp_handler exit");
295         });
296     BMCWEB_LOG_DEBUG("getObjectsWithConnection exit");
297 }
298 
299 /**
300  * @brief Create connections necessary for sensors
301  * @param SensorsAsyncResp Pointer to object holding response data
302  * @param sensorNames Sensors retrieved from chassis
303  * @param callback Callback for processing gathered connections
304  */
305 template <typename Callback>
306 void getConnections(const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
307                     const std::shared_ptr<std::set<std::string>>& sensorNames,
308                     Callback&& callback)
309 {
310     auto objectsWithConnectionCb =
311         [callback = std::forward<Callback>(callback)](
312             const std::set<std::string>& connections,
313             const std::set<std::pair<std::string, std::string>>&
314             /*objectsWithConnection*/) { callback(connections); };
315     getObjectsWithConnection(sensorsAsyncResp, sensorNames,
316                              std::move(objectsWithConnectionCb));
317 }
318 
319 /**
320  * @brief Shrinks the list of sensors for processing
321  * @param SensorsAysncResp  The class holding the Redfish response
322  * @param allSensors  A list of all the sensors associated to the
323  * chassis element (i.e. baseboard, front panel, etc...)
324  * @param activeSensors A list that is a reduction of the incoming
325  * allSensors list.  Eliminate Thermal sensors when a Power request is
326  * made, and eliminate Power sensors when a Thermal request is made.
327  */
328 inline void reduceSensorList(
329     crow::Response& res, std::string_view chassisSubNode,
330     std::span<const std::string_view> sensorTypes,
331     const std::vector<std::string>* allSensors,
332     const std::shared_ptr<std::set<std::string>>& activeSensors)
333 {
334     if ((allSensors == nullptr) || (activeSensors == nullptr))
335     {
336         messages::resourceNotFound(res, chassisSubNode,
337                                    chassisSubNode == sensors::thermalNodeStr
338                                        ? "Temperatures"
339                                        : "Voltages");
340 
341         return;
342     }
343     if (allSensors->empty())
344     {
345         // Nothing to do, the activeSensors object is also empty
346         return;
347     }
348 
349     for (std::string_view type : sensorTypes)
350     {
351         for (const std::string& sensor : *allSensors)
352         {
353             if (sensor.starts_with(type))
354             {
355                 activeSensors->emplace(sensor);
356             }
357         }
358     }
359 }
360 
361 /*
362  *Populates the top level collection for a given subnode.  Populates
363  *SensorCollection, Power, or Thermal schemas.
364  *
365  * */
366 inline void populateChassisNode(nlohmann::json& jsonValue,
367                                 std::string_view chassisSubNode)
368 {
369     if (chassisSubNode == sensors::powerNodeStr)
370     {
371         jsonValue["@odata.type"] = "#Power.v1_5_2.Power";
372     }
373     else if (chassisSubNode == sensors::thermalNodeStr)
374     {
375         jsonValue["@odata.type"] = "#Thermal.v1_4_0.Thermal";
376         jsonValue["Fans"] = nlohmann::json::array();
377         jsonValue["Temperatures"] = nlohmann::json::array();
378     }
379     else if (chassisSubNode == sensors::sensorsNodeStr)
380     {
381         jsonValue["@odata.type"] = "#SensorCollection.SensorCollection";
382         jsonValue["Description"] = "Collection of Sensors for this Chassis";
383         jsonValue["Members"] = nlohmann::json::array();
384         jsonValue["Members@odata.count"] = 0;
385     }
386 
387     if (chassisSubNode != sensors::sensorsNodeStr)
388     {
389         jsonValue["Id"] = chassisSubNode;
390     }
391     jsonValue["Name"] = chassisSubNode;
392 }
393 
394 /**
395  * @brief Retrieves requested chassis sensors and redundancy data from DBus .
396  * @param SensorsAsyncResp   Pointer to object holding response data
397  * @param callback  Callback for next step in gathered sensor processing
398  */
399 template <typename Callback>
400 void getChassis(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
401                 std::string_view chassisId, std::string_view chassisSubNode,
402                 std::span<const std::string_view> sensorTypes,
403                 Callback&& callback)
404 {
405     BMCWEB_LOG_DEBUG("getChassis enter");
406     constexpr std::array<std::string_view, 2> interfaces = {
407         "xyz.openbmc_project.Inventory.Item.Board",
408         "xyz.openbmc_project.Inventory.Item.Chassis"};
409 
410     // Get the Chassis Collection
411     dbus::utility::getSubTreePaths(
412         "/xyz/openbmc_project/inventory", 0, interfaces,
413         [callback = std::forward<Callback>(callback), asyncResp,
414          chassisIdStr{std::string(chassisId)},
415          chassisSubNode{std::string(chassisSubNode)},
416          sensorTypes](const boost::system::error_code& ec,
417                       const dbus::utility::MapperGetSubTreePathsResponse&
418                           chassisPaths) mutable {
419             BMCWEB_LOG_DEBUG("getChassis respHandler enter");
420             if (ec)
421             {
422                 BMCWEB_LOG_ERROR("getChassis respHandler DBUS error: {}", ec);
423                 messages::internalError(asyncResp->res);
424                 return;
425             }
426             const std::string* chassisPath = nullptr;
427             for (const std::string& chassis : chassisPaths)
428             {
429                 sdbusplus::message::object_path path(chassis);
430                 std::string chassisName = path.filename();
431                 if (chassisName.empty())
432                 {
433                     BMCWEB_LOG_ERROR("Failed to find '/' in {}", chassis);
434                     continue;
435                 }
436                 if (chassisName == chassisIdStr)
437                 {
438                     chassisPath = &chassis;
439                     break;
440                 }
441             }
442             if (chassisPath == nullptr)
443             {
444                 messages::resourceNotFound(asyncResp->res, "Chassis",
445                                            chassisIdStr);
446                 return;
447             }
448             populateChassisNode(asyncResp->res.jsonValue, chassisSubNode);
449 
450             asyncResp->res.jsonValue["@odata.id"] = boost::urls::format(
451                 "/redfish/v1/Chassis/{}/{}", chassisIdStr, chassisSubNode);
452 
453             // Get the list of all sensors for this Chassis element
454             std::string sensorPath = *chassisPath + "/all_sensors";
455             dbus::utility::getAssociationEndPoints(
456                 sensorPath, [asyncResp, chassisSubNode, sensorTypes,
457                              callback = std::forward<Callback>(callback)](
458                                 const boost::system::error_code& ec2,
459                                 const dbus::utility::MapperEndPoints&
460                                     nodeSensorList) mutable {
461                     if (ec2)
462                     {
463                         if (ec2.value() != EBADR)
464                         {
465                             messages::internalError(asyncResp->res);
466                             return;
467                         }
468                     }
469                     const std::shared_ptr<std::set<std::string>>
470                         culledSensorList =
471                             std::make_shared<std::set<std::string>>();
472                     reduceSensorList(asyncResp->res, chassisSubNode,
473                                      sensorTypes, &nodeSensorList,
474                                      culledSensorList);
475                     BMCWEB_LOG_DEBUG("Finishing with {}",
476                                      culledSensorList->size());
477                     callback(culledSensorList);
478                 });
479         });
480     BMCWEB_LOG_DEBUG("getChassis exit");
481 }
482 
483 /**
484  * @brief Builds a json sensor representation of a sensor.
485  * @param sensorName  The name of the sensor to be built
486  * @param sensorType  The type (temperature, fan_tach, etc) of the sensor to
487  * build
488  * @param chassisSubNode The subnode (thermal, sensor, etc) of the sensor
489  * @param interfacesDict  A dictionary of the interfaces and properties of said
490  * interfaces to be built from
491  * @param sensorJson  The json object to fill
492  * @param inventoryItem D-Bus inventory item associated with the sensor.  Will
493  * be nullptr if no associated inventory item was found.
494  */
495 inline void objectInterfacesToJson(
496     const std::string& sensorName, const std::string& sensorType,
497     const sensor_utils::ChassisSubNode chassisSubNode,
498     const dbus::utility::DBusInterfacesMap& interfacesDict,
499     nlohmann::json& sensorJson, InventoryItem* inventoryItem)
500 {
501     for (const auto& [interface, valuesDict] : interfacesDict)
502     {
503         sensor_utils::objectPropertiesToJson(
504             sensorName, sensorType, chassisSubNode, valuesDict, sensorJson,
505             inventoryItem);
506     }
507     BMCWEB_LOG_DEBUG("Added sensor {}", sensorName);
508 }
509 
510 inline void populateFanRedundancy(
511     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp)
512 {
513     constexpr std::array<std::string_view, 1> interfaces = {
514         "xyz.openbmc_project.Control.FanRedundancy"};
515     dbus::utility::getSubTree(
516         "/xyz/openbmc_project/control", 2, interfaces,
517         [sensorsAsyncResp](
518             const boost::system::error_code& ec,
519             const dbus::utility::MapperGetSubTreeResponse& resp) {
520             if (ec)
521             {
522                 return; // don't have to have this interface
523             }
524             for (const std::pair<std::string, dbus::utility::MapperServiceMap>&
525                      pathPair : resp)
526             {
527                 const std::string& path = pathPair.first;
528                 const dbus::utility::MapperServiceMap& objDict =
529                     pathPair.second;
530                 if (objDict.empty())
531                 {
532                     continue; // this should be impossible
533                 }
534 
535                 const std::string& owner = objDict.begin()->first;
536                 dbus::utility::getAssociationEndPoints(
537                     path + "/chassis",
538                     [path, owner, sensorsAsyncResp](
539                         const boost::system::error_code& ec2,
540                         const dbus::utility::MapperEndPoints& endpoints) {
541                         if (ec2)
542                         {
543                             return; // if they don't have an association we
544                                     // can't tell what chassis is
545                         }
546                         auto found = std::ranges::find_if(
547                             endpoints,
548                             [sensorsAsyncResp](const std::string& entry) {
549                                 return entry.find(
550                                            sensorsAsyncResp->chassisId) !=
551                                        std::string::npos;
552                             });
553 
554                         if (found == endpoints.end())
555                         {
556                             return;
557                         }
558                         dbus::utility::getAllProperties(
559                             *crow::connections::systemBus, owner, path,
560                             "xyz.openbmc_project.Control.FanRedundancy",
561                             [path, sensorsAsyncResp](
562                                 const boost::system::error_code& ec3,
563                                 const dbus::utility::DBusPropertiesMap& ret) {
564                                 if (ec3)
565                                 {
566                                     return; // don't have to have this
567                                             // interface
568                                 }
569 
570                                 const uint8_t* allowedFailures = nullptr;
571                                 const std::vector<std::string>* collection =
572                                     nullptr;
573                                 const std::string* status = nullptr;
574 
575                                 const bool success =
576                                     sdbusplus::unpackPropertiesNoThrow(
577                                         dbus_utils::UnpackErrorPrinter(), ret,
578                                         "AllowedFailures", allowedFailures,
579                                         "Collection", collection, "Status",
580                                         status);
581 
582                                 if (!success)
583                                 {
584                                     messages::internalError(
585                                         sensorsAsyncResp->asyncResp->res);
586                                     return;
587                                 }
588 
589                                 if (allowedFailures == nullptr ||
590                                     collection == nullptr || status == nullptr)
591                                 {
592                                     BMCWEB_LOG_ERROR(
593                                         "Invalid redundancy interface");
594                                     messages::internalError(
595                                         sensorsAsyncResp->asyncResp->res);
596                                     return;
597                                 }
598 
599                                 sdbusplus::message::object_path objectPath(
600                                     path);
601                                 std::string name = objectPath.filename();
602                                 if (name.empty())
603                                 {
604                                     // this should be impossible
605                                     messages::internalError(
606                                         sensorsAsyncResp->asyncResp->res);
607                                     return;
608                                 }
609                                 std::ranges::replace(name, '_', ' ');
610 
611                                 std::string health;
612 
613                                 if (status->ends_with("Full"))
614                                 {
615                                     health = "OK";
616                                 }
617                                 else if (status->ends_with("Degraded"))
618                                 {
619                                     health = "Warning";
620                                 }
621                                 else
622                                 {
623                                     health = "Critical";
624                                 }
625                                 nlohmann::json::array_t redfishCollection;
626                                 const auto& fanRedfish =
627                                     sensorsAsyncResp->asyncResp->res
628                                         .jsonValue["Fans"];
629                                 for (const std::string& item : *collection)
630                                 {
631                                     sdbusplus::message::object_path itemPath(
632                                         item);
633                                     std::string itemName = itemPath.filename();
634                                     if (itemName.empty())
635                                     {
636                                         continue;
637                                     }
638                                     /*
639                                     todo(ed): merge patch that fixes the names
640                                     std::replace(itemName.begin(),
641                                                  itemName.end(), '_', ' ');*/
642                                     auto schemaItem = std::ranges::find_if(
643                                         fanRedfish,
644                                         [itemName](const nlohmann::json& fan) {
645                                             return fan["Name"] == itemName;
646                                         });
647                                     if (schemaItem != fanRedfish.end())
648                                     {
649                                         nlohmann::json::object_t collectionId;
650                                         collectionId["@odata.id"] =
651                                             (*schemaItem)["@odata.id"];
652                                         redfishCollection.emplace_back(
653                                             std::move(collectionId));
654                                     }
655                                     else
656                                     {
657                                         BMCWEB_LOG_ERROR(
658                                             "failed to find fan in schema");
659                                         messages::internalError(
660                                             sensorsAsyncResp->asyncResp->res);
661                                         return;
662                                     }
663                                 }
664 
665                                 size_t minNumNeeded =
666                                     collection->empty()
667                                         ? 0
668                                         : collection->size() - *allowedFailures;
669                                 nlohmann::json& jResp =
670                                     sensorsAsyncResp->asyncResp->res
671                                         .jsonValue["Redundancy"];
672 
673                                 nlohmann::json::object_t redundancy;
674                                 boost::urls::url url = boost::urls::format(
675                                     "/redfish/v1/Chassis/{}/{}",
676                                     sensorsAsyncResp->chassisId,
677                                     sensorsAsyncResp->chassisSubNode);
678                                 url.set_fragment(
679                                     ("/Redundancy"_json_pointer / jResp.size())
680                                         .to_string());
681                                 redundancy["@odata.id"] = std::move(url);
682                                 redundancy["@odata.type"] =
683                                     "#Redundancy.v1_3_2.Redundancy";
684                                 redundancy["MinNumNeeded"] = minNumNeeded;
685                                 redundancy["Mode"] =
686                                     redundancy::RedundancyType::NPlusM;
687                                 redundancy["Name"] = name;
688                                 redundancy["RedundancySet"] = redfishCollection;
689                                 redundancy["Status"]["Health"] = health;
690                                 redundancy["Status"]["State"] =
691                                     resource::State::Enabled;
692 
693                                 jResp.emplace_back(std::move(redundancy));
694                             });
695                     });
696             }
697         });
698 }
699 
700 inline void
701     sortJSONResponse(const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp)
702 {
703     nlohmann::json& response = sensorsAsyncResp->asyncResp->res.jsonValue;
704     std::array<std::string, 2> sensorHeaders{"Temperatures", "Fans"};
705     if (sensorsAsyncResp->chassisSubNode == sensors::powerNodeStr)
706     {
707         sensorHeaders = {"Voltages", "PowerSupplies"};
708     }
709     for (const std::string& sensorGroup : sensorHeaders)
710     {
711         nlohmann::json::iterator entry = response.find(sensorGroup);
712         if (entry == response.end())
713         {
714             continue;
715         }
716         nlohmann::json::array_t* arr =
717             entry->get_ptr<nlohmann::json::array_t*>();
718         if (arr == nullptr)
719         {
720             continue;
721         }
722         json_util::sortJsonArrayByKey(*arr, "Name");
723 
724         // add the index counts to the end of each entry
725         size_t count = 0;
726         for (nlohmann::json& sensorJson : *entry)
727         {
728             nlohmann::json::iterator odata = sensorJson.find("@odata.id");
729             if (odata == sensorJson.end())
730             {
731                 continue;
732             }
733             std::string* value = odata->get_ptr<std::string*>();
734             if (value != nullptr)
735             {
736                 *value += "/" + std::to_string(count);
737                 sensorJson["MemberId"] = std::to_string(count);
738                 count++;
739                 sensorsAsyncResp->updateUri(sensorJson["Name"], *value);
740             }
741         }
742     }
743 }
744 
745 /**
746  * @brief Finds the inventory item with the specified object path.
747  * @param inventoryItems D-Bus inventory items associated with sensors.
748  * @param invItemObjPath D-Bus object path of inventory item.
749  * @return Inventory item within vector, or nullptr if no match found.
750  */
751 inline InventoryItem* findInventoryItem(
752     const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
753     const std::string& invItemObjPath)
754 {
755     for (InventoryItem& inventoryItem : *inventoryItems)
756     {
757         if (inventoryItem.objectPath == invItemObjPath)
758         {
759             return &inventoryItem;
760         }
761     }
762     return nullptr;
763 }
764 
765 /**
766  * @brief Finds the inventory item associated with the specified sensor.
767  * @param inventoryItems D-Bus inventory items associated with sensors.
768  * @param sensorObjPath D-Bus object path of sensor.
769  * @return Inventory item within vector, or nullptr if no match found.
770  */
771 inline InventoryItem* findInventoryItemForSensor(
772     const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
773     const std::string& sensorObjPath)
774 {
775     for (InventoryItem& inventoryItem : *inventoryItems)
776     {
777         if (inventoryItem.sensors.contains(sensorObjPath))
778         {
779             return &inventoryItem;
780         }
781     }
782     return nullptr;
783 }
784 
785 /**
786  * @brief Finds the inventory item associated with the specified led path.
787  * @param inventoryItems D-Bus inventory items associated with sensors.
788  * @param ledObjPath D-Bus object path of led.
789  * @return Inventory item within vector, or nullptr if no match found.
790  */
791 inline InventoryItem* findInventoryItemForLed(
792     std::vector<InventoryItem>& inventoryItems, const std::string& ledObjPath)
793 {
794     for (InventoryItem& inventoryItem : inventoryItems)
795     {
796         if (inventoryItem.ledObjectPath == ledObjPath)
797         {
798             return &inventoryItem;
799         }
800     }
801     return nullptr;
802 }
803 
804 /**
805  * @brief Adds inventory item and associated sensor to specified vector.
806  *
807  * Adds a new InventoryItem to the vector if necessary.  Searches for an
808  * existing InventoryItem with the specified object path.  If not found, one is
809  * added to the vector.
810  *
811  * Next, the specified sensor is added to the set of sensors associated with the
812  * InventoryItem.
813  *
814  * @param inventoryItems D-Bus inventory items associated with sensors.
815  * @param invItemObjPath D-Bus object path of inventory item.
816  * @param sensorObjPath D-Bus object path of sensor
817  */
818 inline void addInventoryItem(
819     const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
820     const std::string& invItemObjPath, const std::string& sensorObjPath)
821 {
822     // Look for inventory item in vector
823     InventoryItem* inventoryItem =
824         findInventoryItem(inventoryItems, invItemObjPath);
825 
826     // If inventory item doesn't exist in vector, add it
827     if (inventoryItem == nullptr)
828     {
829         inventoryItems->emplace_back(invItemObjPath);
830         inventoryItem = &(inventoryItems->back());
831     }
832 
833     // Add sensor to set of sensors associated with inventory item
834     inventoryItem->sensors.emplace(sensorObjPath);
835 }
836 
837 /**
838  * @brief Stores D-Bus data in the specified inventory item.
839  *
840  * Finds D-Bus data in the specified map of interfaces.  Stores the data in the
841  * specified InventoryItem.
842  *
843  * This data is later used to provide sensor property values in the JSON
844  * response.
845  *
846  * @param inventoryItem Inventory item where data will be stored.
847  * @param interfacesDict Map containing D-Bus interfaces and their properties
848  * for the specified inventory item.
849  */
850 inline void storeInventoryItemData(
851     InventoryItem& inventoryItem,
852     const dbus::utility::DBusInterfacesMap& interfacesDict)
853 {
854     // Get properties from Inventory.Item interface
855 
856     for (const auto& [interface, values] : interfacesDict)
857     {
858         if (interface == "xyz.openbmc_project.Inventory.Item")
859         {
860             for (const auto& [name, dbusValue] : values)
861             {
862                 if (name == "Present")
863                 {
864                     const bool* value = std::get_if<bool>(&dbusValue);
865                     if (value != nullptr)
866                     {
867                         inventoryItem.isPresent = *value;
868                     }
869                 }
870             }
871         }
872         // Check if Inventory.Item.PowerSupply interface is present
873 
874         if (interface == "xyz.openbmc_project.Inventory.Item.PowerSupply")
875         {
876             inventoryItem.isPowerSupply = true;
877         }
878 
879         // Get properties from Inventory.Decorator.Asset interface
880         if (interface == "xyz.openbmc_project.Inventory.Decorator.Asset")
881         {
882             for (const auto& [name, dbusValue] : values)
883             {
884                 if (name == "Manufacturer")
885                 {
886                     const std::string* value =
887                         std::get_if<std::string>(&dbusValue);
888                     if (value != nullptr)
889                     {
890                         inventoryItem.manufacturer = *value;
891                     }
892                 }
893                 if (name == "Model")
894                 {
895                     const std::string* value =
896                         std::get_if<std::string>(&dbusValue);
897                     if (value != nullptr)
898                     {
899                         inventoryItem.model = *value;
900                     }
901                 }
902                 if (name == "SerialNumber")
903                 {
904                     const std::string* value =
905                         std::get_if<std::string>(&dbusValue);
906                     if (value != nullptr)
907                     {
908                         inventoryItem.serialNumber = *value;
909                     }
910                 }
911                 if (name == "PartNumber")
912                 {
913                     const std::string* value =
914                         std::get_if<std::string>(&dbusValue);
915                     if (value != nullptr)
916                     {
917                         inventoryItem.partNumber = *value;
918                     }
919                 }
920             }
921         }
922 
923         if (interface ==
924             "xyz.openbmc_project.State.Decorator.OperationalStatus")
925         {
926             for (const auto& [name, dbusValue] : values)
927             {
928                 if (name == "Functional")
929                 {
930                     const bool* value = std::get_if<bool>(&dbusValue);
931                     if (value != nullptr)
932                     {
933                         inventoryItem.isFunctional = *value;
934                     }
935                 }
936             }
937         }
938     }
939 }
940 
941 /**
942  * @brief Gets D-Bus data for inventory items associated with sensors.
943  *
944  * Uses the specified connections (services) to obtain D-Bus data for inventory
945  * items associated with sensors.  Stores the resulting data in the
946  * inventoryItems vector.
947  *
948  * This data is later used to provide sensor property values in the JSON
949  * response.
950  *
951  * Finds the inventory item data asynchronously.  Invokes callback when data has
952  * been obtained.
953  *
954  * The callback must have the following signature:
955  *   @code
956  *   callback(void)
957  *   @endcode
958  *
959  * This function is called recursively, obtaining data asynchronously from one
960  * connection in each call.  This ensures the callback is not invoked until the
961  * last asynchronous function has completed.
962  *
963  * @param sensorsAsyncResp Pointer to object holding response data.
964  * @param inventoryItems D-Bus inventory items associated with sensors.
965  * @param invConnections Connections that provide data for the inventory items.
966  * implements ObjectManager.
967  * @param callback Callback to invoke when inventory data has been obtained.
968  * @param invConnectionsIndex Current index in invConnections.  Only specified
969  * in recursive calls to this function.
970  */
971 template <typename Callback>
972 void getInventoryItemsData(
973     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
974     const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
975     const std::shared_ptr<std::set<std::string>>& invConnections,
976     Callback&& callback, size_t invConnectionsIndex = 0)
977 {
978     BMCWEB_LOG_DEBUG("getInventoryItemsData enter");
979 
980     // If no more connections left, call callback
981     if (invConnectionsIndex >= invConnections->size())
982     {
983         callback();
984         BMCWEB_LOG_DEBUG("getInventoryItemsData exit");
985         return;
986     }
987 
988     // Get inventory item data from current connection
989     auto it = invConnections->begin();
990     std::advance(it, invConnectionsIndex);
991     if (it != invConnections->end())
992     {
993         const std::string& invConnection = *it;
994 
995         // Get all object paths and their interfaces for current connection
996         sdbusplus::message::object_path path("/xyz/openbmc_project/inventory");
997         dbus::utility::getManagedObjects(
998             invConnection, path,
999             [sensorsAsyncResp, inventoryItems, invConnections,
1000              callback = std::forward<Callback>(callback), invConnectionsIndex](
1001                 const boost::system::error_code& ec,
1002                 const dbus::utility::ManagedObjectType& resp) mutable {
1003                 BMCWEB_LOG_DEBUG("getInventoryItemsData respHandler enter");
1004                 if (ec)
1005                 {
1006                     BMCWEB_LOG_ERROR(
1007                         "getInventoryItemsData respHandler DBus error {}", ec);
1008                     messages::internalError(sensorsAsyncResp->asyncResp->res);
1009                     return;
1010                 }
1011 
1012                 // Loop through returned object paths
1013                 for (const auto& objDictEntry : resp)
1014                 {
1015                     const std::string& objPath =
1016                         static_cast<const std::string&>(objDictEntry.first);
1017 
1018                     // If this object path is one of the specified inventory
1019                     // items
1020                     InventoryItem* inventoryItem =
1021                         findInventoryItem(inventoryItems, objPath);
1022                     if (inventoryItem != nullptr)
1023                     {
1024                         // Store inventory data in InventoryItem
1025                         storeInventoryItemData(*inventoryItem,
1026                                                objDictEntry.second);
1027                     }
1028                 }
1029 
1030                 // Recurse to get inventory item data from next connection
1031                 getInventoryItemsData(sensorsAsyncResp, inventoryItems,
1032                                       invConnections, std::move(callback),
1033                                       invConnectionsIndex + 1);
1034 
1035                 BMCWEB_LOG_DEBUG("getInventoryItemsData respHandler exit");
1036             });
1037     }
1038 
1039     BMCWEB_LOG_DEBUG("getInventoryItemsData exit");
1040 }
1041 
1042 /**
1043  * @brief Gets connections that provide D-Bus data for inventory items.
1044  *
1045  * Gets the D-Bus connections (services) that provide data for the inventory
1046  * items that are associated with sensors.
1047  *
1048  * Finds the connections asynchronously.  Invokes callback when information has
1049  * been obtained.
1050  *
1051  * The callback must have the following signature:
1052  *   @code
1053  *   callback(std::shared_ptr<std::set<std::string>> invConnections)
1054  *   @endcode
1055  *
1056  * @param sensorsAsyncResp Pointer to object holding response data.
1057  * @param inventoryItems D-Bus inventory items associated with sensors.
1058  * @param callback Callback to invoke when connections have been obtained.
1059  */
1060 template <typename Callback>
1061 void getInventoryItemsConnections(
1062     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1063     const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
1064     Callback&& callback)
1065 {
1066     BMCWEB_LOG_DEBUG("getInventoryItemsConnections enter");
1067 
1068     const std::string path = "/xyz/openbmc_project/inventory";
1069     constexpr std::array<std::string_view, 4> interfaces = {
1070         "xyz.openbmc_project.Inventory.Item",
1071         "xyz.openbmc_project.Inventory.Item.PowerSupply",
1072         "xyz.openbmc_project.Inventory.Decorator.Asset",
1073         "xyz.openbmc_project.State.Decorator.OperationalStatus"};
1074 
1075     // Make call to ObjectMapper to find all inventory items
1076     dbus::utility::getSubTree(
1077         path, 0, interfaces,
1078         [callback = std::forward<Callback>(callback), sensorsAsyncResp,
1079          inventoryItems](
1080             const boost::system::error_code& ec,
1081             const dbus::utility::MapperGetSubTreeResponse& subtree) mutable {
1082             // Response handler for parsing output from GetSubTree
1083             BMCWEB_LOG_DEBUG("getInventoryItemsConnections respHandler enter");
1084             if (ec)
1085             {
1086                 messages::internalError(sensorsAsyncResp->asyncResp->res);
1087                 BMCWEB_LOG_ERROR(
1088                     "getInventoryItemsConnections respHandler DBus error {}",
1089                     ec);
1090                 return;
1091             }
1092 
1093             // Make unique list of connections for desired inventory items
1094             std::shared_ptr<std::set<std::string>> invConnections =
1095                 std::make_shared<std::set<std::string>>();
1096 
1097             // Loop through objects from GetSubTree
1098             for (const std::pair<std::string,
1099                                  std::vector<std::pair<
1100                                      std::string, std::vector<std::string>>>>&
1101                      object : subtree)
1102             {
1103                 // Check if object path is one of the specified inventory items
1104                 const std::string& objPath = object.first;
1105                 if (findInventoryItem(inventoryItems, objPath) != nullptr)
1106                 {
1107                     // Store all connections to inventory item
1108                     for (const std::pair<std::string, std::vector<std::string>>&
1109                              objData : object.second)
1110                     {
1111                         const std::string& invConnection = objData.first;
1112                         invConnections->insert(invConnection);
1113                     }
1114                 }
1115             }
1116 
1117             callback(invConnections);
1118             BMCWEB_LOG_DEBUG("getInventoryItemsConnections respHandler exit");
1119         });
1120     BMCWEB_LOG_DEBUG("getInventoryItemsConnections exit");
1121 }
1122 
1123 /**
1124  * @brief Gets associations from sensors to inventory items.
1125  *
1126  * Looks for ObjectMapper associations from the specified sensors to related
1127  * inventory items. Then finds the associations from those inventory items to
1128  * their LEDs, if any.
1129  *
1130  * Finds the inventory items asynchronously.  Invokes callback when information
1131  * has been obtained.
1132  *
1133  * The callback must have the following signature:
1134  *   @code
1135  *   callback(std::shared_ptr<std::vector<InventoryItem>> inventoryItems)
1136  *   @endcode
1137  *
1138  * @param sensorsAsyncResp Pointer to object holding response data.
1139  * @param sensorNames All sensors within the current chassis.
1140  * implements ObjectManager.
1141  * @param callback Callback to invoke when inventory items have been obtained.
1142  */
1143 template <typename Callback>
1144 void getInventoryItemAssociations(
1145     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1146     const std::shared_ptr<std::set<std::string>>& sensorNames,
1147     Callback&& callback)
1148 {
1149     BMCWEB_LOG_DEBUG("getInventoryItemAssociations enter");
1150 
1151     // Call GetManagedObjects on the ObjectMapper to get all associations
1152     sdbusplus::message::object_path path("/");
1153     dbus::utility::getManagedObjects(
1154         "xyz.openbmc_project.ObjectMapper", path,
1155         [callback = std::forward<Callback>(callback), sensorsAsyncResp,
1156          sensorNames](const boost::system::error_code& ec,
1157                       const dbus::utility::ManagedObjectType& resp) mutable {
1158             BMCWEB_LOG_DEBUG("getInventoryItemAssociations respHandler enter");
1159             if (ec)
1160             {
1161                 BMCWEB_LOG_ERROR(
1162                     "getInventoryItemAssociations respHandler DBus error {}",
1163                     ec);
1164                 messages::internalError(sensorsAsyncResp->asyncResp->res);
1165                 return;
1166             }
1167 
1168             // Create vector to hold list of inventory items
1169             std::shared_ptr<std::vector<InventoryItem>> inventoryItems =
1170                 std::make_shared<std::vector<InventoryItem>>();
1171 
1172             // Loop through returned object paths
1173             std::string sensorAssocPath;
1174             sensorAssocPath.reserve(128); // avoid memory allocations
1175             for (const auto& objDictEntry : resp)
1176             {
1177                 const std::string& objPath =
1178                     static_cast<const std::string&>(objDictEntry.first);
1179 
1180                 // If path is inventory association for one of the specified
1181                 // sensors
1182                 for (const std::string& sensorName : *sensorNames)
1183                 {
1184                     sensorAssocPath = sensorName;
1185                     sensorAssocPath += "/inventory";
1186                     if (objPath == sensorAssocPath)
1187                     {
1188                         // Get Association interface for object path
1189                         for (const auto& [interface, values] :
1190                              objDictEntry.second)
1191                         {
1192                             if (interface == "xyz.openbmc_project.Association")
1193                             {
1194                                 for (const auto& [valueName, value] : values)
1195                                 {
1196                                     if (valueName == "endpoints")
1197                                     {
1198                                         const std::vector<std::string>*
1199                                             endpoints = std::get_if<
1200                                                 std::vector<std::string>>(
1201                                                 &value);
1202                                         if ((endpoints != nullptr) &&
1203                                             !endpoints->empty())
1204                                         {
1205                                             // Add inventory item to vector
1206                                             const std::string& invItemPath =
1207                                                 endpoints->front();
1208                                             addInventoryItem(inventoryItems,
1209                                                              invItemPath,
1210                                                              sensorName);
1211                                         }
1212                                     }
1213                                 }
1214                             }
1215                         }
1216                         break;
1217                     }
1218                 }
1219             }
1220 
1221             // Now loop through the returned object paths again, this time to
1222             // find the leds associated with the inventory items we just found
1223             std::string inventoryAssocPath;
1224             inventoryAssocPath.reserve(128); // avoid memory allocations
1225             for (const auto& objDictEntry : resp)
1226             {
1227                 const std::string& objPath =
1228                     static_cast<const std::string&>(objDictEntry.first);
1229 
1230                 for (InventoryItem& inventoryItem : *inventoryItems)
1231                 {
1232                     inventoryAssocPath = inventoryItem.objectPath;
1233                     inventoryAssocPath += "/leds";
1234                     if (objPath == inventoryAssocPath)
1235                     {
1236                         for (const auto& [interface, values] :
1237                              objDictEntry.second)
1238                         {
1239                             if (interface == "xyz.openbmc_project.Association")
1240                             {
1241                                 for (const auto& [valueName, value] : values)
1242                                 {
1243                                     if (valueName == "endpoints")
1244                                     {
1245                                         const std::vector<std::string>*
1246                                             endpoints = std::get_if<
1247                                                 std::vector<std::string>>(
1248                                                 &value);
1249                                         if ((endpoints != nullptr) &&
1250                                             !endpoints->empty())
1251                                         {
1252                                             // Add inventory item to vector
1253                                             // Store LED path in inventory item
1254                                             const std::string& ledPath =
1255                                                 endpoints->front();
1256                                             inventoryItem.ledObjectPath =
1257                                                 ledPath;
1258                                         }
1259                                     }
1260                                 }
1261                             }
1262                         }
1263 
1264                         break;
1265                     }
1266                 }
1267             }
1268             callback(inventoryItems);
1269             BMCWEB_LOG_DEBUG("getInventoryItemAssociations respHandler exit");
1270         });
1271 
1272     BMCWEB_LOG_DEBUG("getInventoryItemAssociations exit");
1273 }
1274 
1275 /**
1276  * @brief Gets D-Bus data for inventory item leds associated with sensors.
1277  *
1278  * Uses the specified connections (services) to obtain D-Bus data for inventory
1279  * item leds associated with sensors.  Stores the resulting data in the
1280  * inventoryItems vector.
1281  *
1282  * This data is later used to provide sensor property values in the JSON
1283  * response.
1284  *
1285  * Finds the inventory item led data asynchronously.  Invokes callback when data
1286  * has been obtained.
1287  *
1288  * The callback must have the following signature:
1289  *   @code
1290  *   callback()
1291  *   @endcode
1292  *
1293  * This function is called recursively, obtaining data asynchronously from one
1294  * connection in each call.  This ensures the callback is not invoked until the
1295  * last asynchronous function has completed.
1296  *
1297  * @param sensorsAsyncResp Pointer to object holding response data.
1298  * @param inventoryItems D-Bus inventory items associated with sensors.
1299  * @param ledConnections Connections that provide data for the inventory leds.
1300  * @param callback Callback to invoke when inventory data has been obtained.
1301  * @param ledConnectionsIndex Current index in ledConnections.  Only specified
1302  * in recursive calls to this function.
1303  */
1304 template <typename Callback>
1305 void getInventoryLedData(
1306     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1307     const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
1308     const std::shared_ptr<std::map<std::string, std::string>>& ledConnections,
1309     Callback&& callback, size_t ledConnectionsIndex = 0)
1310 {
1311     BMCWEB_LOG_DEBUG("getInventoryLedData enter");
1312 
1313     // If no more connections left, call callback
1314     if (ledConnectionsIndex >= ledConnections->size())
1315     {
1316         callback();
1317         BMCWEB_LOG_DEBUG("getInventoryLedData exit");
1318         return;
1319     }
1320 
1321     // Get inventory item data from current connection
1322     auto it = ledConnections->begin();
1323     std::advance(it, ledConnectionsIndex);
1324     if (it != ledConnections->end())
1325     {
1326         const std::string& ledPath = (*it).first;
1327         const std::string& ledConnection = (*it).second;
1328         // Response handler for Get State property
1329         auto respHandler =
1330             [sensorsAsyncResp, inventoryItems, ledConnections, ledPath,
1331              callback = std::forward<Callback>(callback),
1332              ledConnectionsIndex](const boost::system::error_code& ec,
1333                                   const std::string& state) mutable {
1334                 BMCWEB_LOG_DEBUG("getInventoryLedData respHandler enter");
1335                 if (ec)
1336                 {
1337                     BMCWEB_LOG_ERROR(
1338                         "getInventoryLedData respHandler DBus error {}", ec);
1339                     messages::internalError(sensorsAsyncResp->asyncResp->res);
1340                     return;
1341                 }
1342 
1343                 BMCWEB_LOG_DEBUG("Led state: {}", state);
1344                 // Find inventory item with this LED object path
1345                 InventoryItem* inventoryItem =
1346                     findInventoryItemForLed(*inventoryItems, ledPath);
1347                 if (inventoryItem != nullptr)
1348                 {
1349                     // Store LED state in InventoryItem
1350                     if (state.ends_with("On"))
1351                     {
1352                         inventoryItem->ledState = sensor_utils::LedState::ON;
1353                     }
1354                     else if (state.ends_with("Blink"))
1355                     {
1356                         inventoryItem->ledState = sensor_utils::LedState::BLINK;
1357                     }
1358                     else if (state.ends_with("Off"))
1359                     {
1360                         inventoryItem->ledState = sensor_utils::LedState::OFF;
1361                     }
1362                     else
1363                     {
1364                         inventoryItem->ledState =
1365                             sensor_utils::LedState::UNKNOWN;
1366                     }
1367                 }
1368 
1369                 // Recurse to get LED data from next connection
1370                 getInventoryLedData(sensorsAsyncResp, inventoryItems,
1371                                     ledConnections, std::move(callback),
1372                                     ledConnectionsIndex + 1);
1373 
1374                 BMCWEB_LOG_DEBUG("getInventoryLedData respHandler exit");
1375             };
1376 
1377         // Get the State property for the current LED
1378         dbus::utility::getProperty<std::string>(
1379             ledConnection, ledPath, "xyz.openbmc_project.Led.Physical", "State",
1380             std::move(respHandler));
1381     }
1382 
1383     BMCWEB_LOG_DEBUG("getInventoryLedData exit");
1384 }
1385 
1386 /**
1387  * @brief Gets LED data for LEDs associated with given inventory items.
1388  *
1389  * Gets the D-Bus connections (services) that provide LED data for the LEDs
1390  * associated with the specified inventory items.  Then gets the LED data from
1391  * each connection and stores it in the inventory item.
1392  *
1393  * This data is later used to provide sensor property values in the JSON
1394  * response.
1395  *
1396  * Finds the LED data asynchronously.  Invokes callback when information has
1397  * been obtained.
1398  *
1399  * The callback must have the following signature:
1400  *   @code
1401  *   callback()
1402  *   @endcode
1403  *
1404  * @param sensorsAsyncResp Pointer to object holding response data.
1405  * @param inventoryItems D-Bus inventory items associated with sensors.
1406  * @param callback Callback to invoke when inventory items have been obtained.
1407  */
1408 template <typename Callback>
1409 void getInventoryLeds(
1410     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1411     const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
1412     Callback&& callback)
1413 {
1414     BMCWEB_LOG_DEBUG("getInventoryLeds enter");
1415 
1416     const std::string path = "/xyz/openbmc_project";
1417     constexpr std::array<std::string_view, 1> interfaces = {
1418         "xyz.openbmc_project.Led.Physical"};
1419 
1420     // Make call to ObjectMapper to find all inventory items
1421     dbus::utility::getSubTree(
1422         path, 0, interfaces,
1423         [callback = std::forward<Callback>(callback), sensorsAsyncResp,
1424          inventoryItems](
1425             const boost::system::error_code& ec,
1426             const dbus::utility::MapperGetSubTreeResponse& subtree) mutable {
1427             // Response handler for parsing output from GetSubTree
1428             BMCWEB_LOG_DEBUG("getInventoryLeds respHandler enter");
1429             if (ec)
1430             {
1431                 messages::internalError(sensorsAsyncResp->asyncResp->res);
1432                 BMCWEB_LOG_ERROR("getInventoryLeds respHandler DBus error {}",
1433                                  ec);
1434                 return;
1435             }
1436 
1437             // Build map of LED object paths to connections
1438             std::shared_ptr<std::map<std::string, std::string>> ledConnections =
1439                 std::make_shared<std::map<std::string, std::string>>();
1440 
1441             // Loop through objects from GetSubTree
1442             for (const std::pair<std::string,
1443                                  std::vector<std::pair<
1444                                      std::string, std::vector<std::string>>>>&
1445                      object : subtree)
1446             {
1447                 // Check if object path is LED for one of the specified
1448                 // inventory items
1449                 const std::string& ledPath = object.first;
1450                 if (findInventoryItemForLed(*inventoryItems, ledPath) !=
1451                     nullptr)
1452                 {
1453                     // Add mapping from ledPath to connection
1454                     const std::string& connection =
1455                         object.second.begin()->first;
1456                     (*ledConnections)[ledPath] = connection;
1457                     BMCWEB_LOG_DEBUG("Added mapping {} -> {}", ledPath,
1458                                      connection);
1459                 }
1460             }
1461 
1462             getInventoryLedData(sensorsAsyncResp, inventoryItems,
1463                                 ledConnections, std::move(callback));
1464             BMCWEB_LOG_DEBUG("getInventoryLeds respHandler exit");
1465         });
1466     BMCWEB_LOG_DEBUG("getInventoryLeds exit");
1467 }
1468 
1469 /**
1470  * @brief Gets D-Bus data for Power Supply Attributes such as EfficiencyPercent
1471  *
1472  * Uses the specified connections (services) (currently assumes just one) to
1473  * obtain D-Bus data for Power Supply Attributes. Stores the resulting data in
1474  * the inventoryItems vector. Only stores data in Power Supply inventoryItems.
1475  *
1476  * This data is later used to provide sensor property values in the JSON
1477  * response.
1478  *
1479  * Finds the Power Supply Attributes data asynchronously.  Invokes callback
1480  * when data has been obtained.
1481  *
1482  * The callback must have the following signature:
1483  *   @code
1484  *   callback(std::shared_ptr<std::vector<InventoryItem>> inventoryItems)
1485  *   @endcode
1486  *
1487  * @param sensorsAsyncResp Pointer to object holding response data.
1488  * @param inventoryItems D-Bus inventory items associated with sensors.
1489  * @param psAttributesConnections Connections that provide data for the Power
1490  *        Supply Attributes
1491  * @param callback Callback to invoke when data has been obtained.
1492  */
1493 template <typename Callback>
1494 void getPowerSupplyAttributesData(
1495     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1496     const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
1497     const std::map<std::string, std::string>& psAttributesConnections,
1498     Callback&& callback)
1499 {
1500     BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData enter");
1501 
1502     if (psAttributesConnections.empty())
1503     {
1504         BMCWEB_LOG_DEBUG("Can't find PowerSupplyAttributes, no connections!");
1505         callback(inventoryItems);
1506         return;
1507     }
1508 
1509     // Assuming just one connection (service) for now
1510     auto it = psAttributesConnections.begin();
1511 
1512     const std::string& psAttributesPath = (*it).first;
1513     const std::string& psAttributesConnection = (*it).second;
1514 
1515     // Response handler for Get DeratingFactor property
1516     auto respHandler = [sensorsAsyncResp, inventoryItems,
1517                         callback = std::forward<Callback>(callback)](
1518                            const boost::system::error_code& ec,
1519                            uint32_t value) mutable {
1520         BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData respHandler enter");
1521         if (ec)
1522         {
1523             BMCWEB_LOG_ERROR(
1524                 "getPowerSupplyAttributesData respHandler DBus error {}", ec);
1525             messages::internalError(sensorsAsyncResp->asyncResp->res);
1526             return;
1527         }
1528 
1529         BMCWEB_LOG_DEBUG("PS EfficiencyPercent value: {}", value);
1530         // Store value in Power Supply Inventory Items
1531         for (InventoryItem& inventoryItem : *inventoryItems)
1532         {
1533             if (inventoryItem.isPowerSupply)
1534             {
1535                 inventoryItem.powerSupplyEfficiencyPercent =
1536                     static_cast<int>(value);
1537             }
1538         }
1539 
1540         BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData respHandler exit");
1541         callback(inventoryItems);
1542     };
1543 
1544     // Get the DeratingFactor property for the PowerSupplyAttributes
1545     // Currently only property on the interface/only one we care about
1546     dbus::utility::getProperty<uint32_t>(
1547         psAttributesConnection, psAttributesPath,
1548         "xyz.openbmc_project.Control.PowerSupplyAttributes", "DeratingFactor",
1549         std::move(respHandler));
1550 
1551     BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData exit");
1552 }
1553 
1554 /**
1555  * @brief Gets the Power Supply Attributes such as EfficiencyPercent
1556  *
1557  * Gets the D-Bus connection (service) that provides Power Supply Attributes
1558  * data. Then gets the Power Supply Attributes data from the connection
1559  * (currently just assumes 1 connection) and stores the data in the inventory
1560  * item.
1561  *
1562  * This data is later used to provide sensor property values in the JSON
1563  * response. DeratingFactor on D-Bus is mapped to EfficiencyPercent on Redfish.
1564  *
1565  * Finds the Power Supply Attributes data asynchronously. Invokes callback
1566  * when information has been obtained.
1567  *
1568  * The callback must have the following signature:
1569  *   @code
1570  *   callback(std::shared_ptr<std::vector<InventoryItem>> inventoryItems)
1571  *   @endcode
1572  *
1573  * @param sensorsAsyncResp Pointer to object holding response data.
1574  * @param inventoryItems D-Bus inventory items associated with sensors.
1575  * @param callback Callback to invoke when data has been obtained.
1576  */
1577 template <typename Callback>
1578 void getPowerSupplyAttributes(
1579     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1580     const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
1581     Callback&& callback)
1582 {
1583     BMCWEB_LOG_DEBUG("getPowerSupplyAttributes enter");
1584 
1585     // Only need the power supply attributes when the Power Schema
1586     if (sensorsAsyncResp->chassisSubNode != sensors::powerNodeStr)
1587     {
1588         BMCWEB_LOG_DEBUG("getPowerSupplyAttributes exit since not Power");
1589         callback(inventoryItems);
1590         return;
1591     }
1592 
1593     constexpr std::array<std::string_view, 1> interfaces = {
1594         "xyz.openbmc_project.Control.PowerSupplyAttributes"};
1595 
1596     // Make call to ObjectMapper to find the PowerSupplyAttributes service
1597     dbus::utility::getSubTree(
1598         "/xyz/openbmc_project", 0, interfaces,
1599         [callback = std::forward<Callback>(callback), sensorsAsyncResp,
1600          inventoryItems](
1601             const boost::system::error_code& ec,
1602             const dbus::utility::MapperGetSubTreeResponse& subtree) mutable {
1603             // Response handler for parsing output from GetSubTree
1604             BMCWEB_LOG_DEBUG("getPowerSupplyAttributes respHandler enter");
1605             if (ec)
1606             {
1607                 messages::internalError(sensorsAsyncResp->asyncResp->res);
1608                 BMCWEB_LOG_ERROR(
1609                     "getPowerSupplyAttributes respHandler DBus error {}", ec);
1610                 return;
1611             }
1612             if (subtree.empty())
1613             {
1614                 BMCWEB_LOG_DEBUG("Can't find Power Supply Attributes!");
1615                 callback(inventoryItems);
1616                 return;
1617             }
1618 
1619             // Currently we only support 1 power supply attribute, use this for
1620             // all the power supplies. Build map of object path to connection.
1621             // Assume just 1 connection and 1 path for now.
1622             std::map<std::string, std::string> psAttributesConnections;
1623 
1624             if (subtree[0].first.empty() || subtree[0].second.empty())
1625             {
1626                 BMCWEB_LOG_DEBUG("Power Supply Attributes mapper error!");
1627                 callback(inventoryItems);
1628                 return;
1629             }
1630 
1631             const std::string& psAttributesPath = subtree[0].first;
1632             const std::string& connection = subtree[0].second.begin()->first;
1633 
1634             if (connection.empty())
1635             {
1636                 BMCWEB_LOG_DEBUG("Power Supply Attributes mapper error!");
1637                 callback(inventoryItems);
1638                 return;
1639             }
1640 
1641             psAttributesConnections[psAttributesPath] = connection;
1642             BMCWEB_LOG_DEBUG("Added mapping {} -> {}", psAttributesPath,
1643                              connection);
1644 
1645             getPowerSupplyAttributesData(sensorsAsyncResp, inventoryItems,
1646                                          psAttributesConnections,
1647                                          std::move(callback));
1648             BMCWEB_LOG_DEBUG("getPowerSupplyAttributes respHandler exit");
1649         });
1650     BMCWEB_LOG_DEBUG("getPowerSupplyAttributes exit");
1651 }
1652 
1653 /**
1654  * @brief Gets inventory items associated with sensors.
1655  *
1656  * Finds the inventory items that are associated with the specified sensors.
1657  * Then gets D-Bus data for the inventory items, such as presence and VPD.
1658  *
1659  * This data is later used to provide sensor property values in the JSON
1660  * response.
1661  *
1662  * Finds the inventory items asynchronously.  Invokes callback when the
1663  * inventory items have been obtained.
1664  *
1665  * The callback must have the following signature:
1666  *   @code
1667  *   callback(std::shared_ptr<std::vector<InventoryItem>> inventoryItems)
1668  *   @endcode
1669  *
1670  * @param sensorsAsyncResp Pointer to object holding response data.
1671  * @param sensorNames All sensors within the current chassis.
1672  * implements ObjectManager.
1673  * @param callback Callback to invoke when inventory items have been obtained.
1674  */
1675 template <typename Callback>
1676 inline void
1677     getInventoryItems(const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1678                       const std::shared_ptr<std::set<std::string>>& sensorNames,
1679                       Callback&& callback)
1680 {
1681     BMCWEB_LOG_DEBUG("getInventoryItems enter");
1682     auto getInventoryItemAssociationsCb =
1683         [sensorsAsyncResp, callback = std::forward<Callback>(callback)](
1684             const std::shared_ptr<std::vector<InventoryItem>>&
1685                 inventoryItems) mutable {
1686             BMCWEB_LOG_DEBUG("getInventoryItemAssociationsCb enter");
1687             auto getInventoryItemsConnectionsCb =
1688                 [sensorsAsyncResp, inventoryItems,
1689                  callback = std::forward<Callback>(callback)](
1690                     const std::shared_ptr<std::set<std::string>>&
1691                         invConnections) mutable {
1692                     BMCWEB_LOG_DEBUG("getInventoryItemsConnectionsCb enter");
1693                     auto getInventoryItemsDataCb =
1694                         [sensorsAsyncResp, inventoryItems,
1695                          callback =
1696                              std::forward<Callback>(callback)]() mutable {
1697                             BMCWEB_LOG_DEBUG("getInventoryItemsDataCb enter");
1698 
1699                             auto getInventoryLedsCb =
1700                                 [sensorsAsyncResp, inventoryItems,
1701                                  callback = std::forward<Callback>(
1702                                      callback)]() mutable {
1703                                     BMCWEB_LOG_DEBUG(
1704                                         "getInventoryLedsCb enter");
1705                                     // Find Power Supply Attributes and get the
1706                                     // data
1707                                     getPowerSupplyAttributes(
1708                                         sensorsAsyncResp, inventoryItems,
1709                                         std::move(callback));
1710                                     BMCWEB_LOG_DEBUG("getInventoryLedsCb exit");
1711                                 };
1712 
1713                             // Find led connections and get the data
1714                             getInventoryLeds(sensorsAsyncResp, inventoryItems,
1715                                              std::move(getInventoryLedsCb));
1716                             BMCWEB_LOG_DEBUG("getInventoryItemsDataCb exit");
1717                         };
1718 
1719                     // Get inventory item data from connections
1720                     getInventoryItemsData(sensorsAsyncResp, inventoryItems,
1721                                           invConnections,
1722                                           std::move(getInventoryItemsDataCb));
1723                     BMCWEB_LOG_DEBUG("getInventoryItemsConnectionsCb exit");
1724                 };
1725 
1726             // Get connections that provide inventory item data
1727             getInventoryItemsConnections(
1728                 sensorsAsyncResp, inventoryItems,
1729                 std::move(getInventoryItemsConnectionsCb));
1730             BMCWEB_LOG_DEBUG("getInventoryItemAssociationsCb exit");
1731         };
1732 
1733     // Get associations from sensors to inventory items
1734     getInventoryItemAssociations(sensorsAsyncResp, sensorNames,
1735                                  std::move(getInventoryItemAssociationsCb));
1736     BMCWEB_LOG_DEBUG("getInventoryItems exit");
1737 }
1738 
1739 /**
1740  * @brief Returns JSON PowerSupply object for the specified inventory item.
1741  *
1742  * Searches for a JSON PowerSupply object that matches the specified inventory
1743  * item.  If one is not found, a new PowerSupply object is added to the JSON
1744  * array.
1745  *
1746  * Multiple sensors are often associated with one power supply inventory item.
1747  * As a result, multiple sensor values are stored in one JSON PowerSupply
1748  * object.
1749  *
1750  * @param powerSupplyArray JSON array containing Redfish PowerSupply objects.
1751  * @param inventoryItem Inventory item for the power supply.
1752  * @param chassisId Chassis that contains the power supply.
1753  * @return JSON PowerSupply object for the specified inventory item.
1754  */
1755 inline nlohmann::json& getPowerSupply(nlohmann::json& powerSupplyArray,
1756                                       const InventoryItem& inventoryItem,
1757                                       const std::string& chassisId)
1758 {
1759     std::string nameS;
1760     nameS.resize(inventoryItem.name.size());
1761     std::ranges::replace_copy(inventoryItem.name, nameS.begin(), '_', ' ');
1762     // Check if matching PowerSupply object already exists in JSON array
1763     for (nlohmann::json& powerSupply : powerSupplyArray)
1764     {
1765         nlohmann::json::iterator nameIt = powerSupply.find("Name");
1766         if (nameIt == powerSupply.end())
1767         {
1768             continue;
1769         }
1770         const std::string* name = nameIt->get_ptr<std::string*>();
1771         if (name == nullptr)
1772         {
1773             continue;
1774         }
1775         if (nameS == *name)
1776         {
1777             return powerSupply;
1778         }
1779     }
1780 
1781     // Add new PowerSupply object to JSON array
1782     powerSupplyArray.push_back({});
1783     nlohmann::json& powerSupply = powerSupplyArray.back();
1784     boost::urls::url url =
1785         boost::urls::format("/redfish/v1/Chassis/{}/Power", chassisId);
1786     url.set_fragment(("/PowerSupplies"_json_pointer).to_string());
1787     powerSupply["@odata.id"] = std::move(url);
1788     std::string escaped;
1789     escaped.resize(inventoryItem.name.size());
1790     std::ranges::replace_copy(inventoryItem.name, escaped.begin(), '_', ' ');
1791     powerSupply["Name"] = std::move(escaped);
1792     powerSupply["Manufacturer"] = inventoryItem.manufacturer;
1793     powerSupply["Model"] = inventoryItem.model;
1794     powerSupply["PartNumber"] = inventoryItem.partNumber;
1795     powerSupply["SerialNumber"] = inventoryItem.serialNumber;
1796     sensor_utils::setLedState(powerSupply, &inventoryItem);
1797 
1798     if (inventoryItem.powerSupplyEfficiencyPercent >= 0)
1799     {
1800         powerSupply["EfficiencyPercent"] =
1801             inventoryItem.powerSupplyEfficiencyPercent;
1802     }
1803 
1804     powerSupply["Status"]["State"] =
1805         sensor_utils::getState(&inventoryItem, true);
1806     const char* health = inventoryItem.isFunctional ? "OK" : "Critical";
1807     powerSupply["Status"]["Health"] = health;
1808 
1809     return powerSupply;
1810 }
1811 
1812 /**
1813  * @brief Gets the values of the specified sensors.
1814  *
1815  * Stores the results as JSON in the SensorsAsyncResp.
1816  *
1817  * Gets the sensor values asynchronously.  Stores the results later when the
1818  * information has been obtained.
1819  *
1820  * The sensorNames set contains all requested sensors for the current chassis.
1821  *
1822  * To minimize the number of DBus calls, the DBus method
1823  * org.freedesktop.DBus.ObjectManager.GetManagedObjects() is used to get the
1824  * values of all sensors provided by a connection (service).
1825  *
1826  * The connections set contains all the connections that provide sensor values.
1827  *
1828  * The InventoryItem vector contains D-Bus inventory items associated with the
1829  * sensors.  Inventory item data is needed for some Redfish sensor properties.
1830  *
1831  * @param SensorsAsyncResp Pointer to object holding response data.
1832  * @param sensorNames All requested sensors within the current chassis.
1833  * @param connections Connections that provide sensor values.
1834  * implements ObjectManager.
1835  * @param inventoryItems Inventory items associated with the sensors.
1836  */
1837 inline void getSensorData(
1838     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1839     const std::shared_ptr<std::set<std::string>>& sensorNames,
1840     const std::set<std::string>& connections,
1841     const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems)
1842 {
1843     BMCWEB_LOG_DEBUG("getSensorData enter");
1844     // Get managed objects from all services exposing sensors
1845     for (const std::string& connection : connections)
1846     {
1847         sdbusplus::message::object_path sensorPath(
1848             "/xyz/openbmc_project/sensors");
1849         dbus::utility::getManagedObjects(
1850             connection, sensorPath,
1851             [sensorsAsyncResp, sensorNames,
1852              inventoryItems](const boost::system::error_code& ec,
1853                              const dbus::utility::ManagedObjectType& resp) {
1854                 BMCWEB_LOG_DEBUG("getManagedObjectsCb enter");
1855                 if (ec)
1856                 {
1857                     BMCWEB_LOG_ERROR("getManagedObjectsCb DBUS error: {}", ec);
1858                     messages::internalError(sensorsAsyncResp->asyncResp->res);
1859                     return;
1860                 }
1861                 auto chassisSubNode = sensor_utils::chassisSubNodeFromString(
1862                     sensorsAsyncResp->chassisSubNode);
1863                 // Go through all objects and update response with sensor data
1864                 for (const auto& objDictEntry : resp)
1865                 {
1866                     const std::string& objPath =
1867                         static_cast<const std::string&>(objDictEntry.first);
1868                     BMCWEB_LOG_DEBUG("getManagedObjectsCb parsing object {}",
1869                                      objPath);
1870 
1871                     std::vector<std::string> split;
1872                     // Reserve space for
1873                     // /xyz/openbmc_project/sensors/<name>/<subname>
1874                     split.reserve(6);
1875                     // NOLINTNEXTLINE
1876                     bmcweb::split(split, objPath, '/');
1877                     if (split.size() < 6)
1878                     {
1879                         BMCWEB_LOG_ERROR("Got path that isn't long enough {}",
1880                                          objPath);
1881                         continue;
1882                     }
1883                     // These indexes aren't intuitive, as split puts an empty
1884                     // string at the beginning
1885                     const std::string& sensorType = split[4];
1886                     const std::string& sensorName = split[5];
1887                     BMCWEB_LOG_DEBUG("sensorName {} sensorType {}", sensorName,
1888                                      sensorType);
1889                     if (sensorNames->find(objPath) == sensorNames->end())
1890                     {
1891                         BMCWEB_LOG_DEBUG("{} not in sensor list ", sensorName);
1892                         continue;
1893                     }
1894 
1895                     // Find inventory item (if any) associated with sensor
1896                     InventoryItem* inventoryItem =
1897                         findInventoryItemForSensor(inventoryItems, objPath);
1898 
1899                     const std::string& sensorSchema =
1900                         sensorsAsyncResp->chassisSubNode;
1901 
1902                     nlohmann::json* sensorJson = nullptr;
1903 
1904                     if (sensorSchema == sensors::sensorsNodeStr &&
1905                         !sensorsAsyncResp->efficientExpand)
1906                     {
1907                         std::string sensorId =
1908                             redfish::sensor_utils::getSensorId(sensorName,
1909                                                                sensorType);
1910 
1911                         sensorsAsyncResp->asyncResp->res
1912                             .jsonValue["@odata.id"] = boost::urls::format(
1913                             "/redfish/v1/Chassis/{}/{}/{}",
1914                             sensorsAsyncResp->chassisId,
1915                             sensorsAsyncResp->chassisSubNode, sensorId);
1916                         sensorJson =
1917                             &(sensorsAsyncResp->asyncResp->res.jsonValue);
1918                     }
1919                     else
1920                     {
1921                         std::string fieldName;
1922                         if (sensorsAsyncResp->efficientExpand)
1923                         {
1924                             fieldName = "Members";
1925                         }
1926                         else if (sensorType == "temperature")
1927                         {
1928                             fieldName = "Temperatures";
1929                         }
1930                         else if (sensorType == "fan" ||
1931                                  sensorType == "fan_tach" ||
1932                                  sensorType == "fan_pwm")
1933                         {
1934                             fieldName = "Fans";
1935                         }
1936                         else if (sensorType == "voltage")
1937                         {
1938                             fieldName = "Voltages";
1939                         }
1940                         else if (sensorType == "power")
1941                         {
1942                             if (sensorName == "total_power")
1943                             {
1944                                 fieldName = "PowerControl";
1945                             }
1946                             else if ((inventoryItem != nullptr) &&
1947                                      (inventoryItem->isPowerSupply))
1948                             {
1949                                 fieldName = "PowerSupplies";
1950                             }
1951                             else
1952                             {
1953                                 // Other power sensors are in SensorCollection
1954                                 continue;
1955                             }
1956                         }
1957                         else
1958                         {
1959                             BMCWEB_LOG_ERROR(
1960                                 "Unsure how to handle sensorType {}",
1961                                 sensorType);
1962                             continue;
1963                         }
1964 
1965                         nlohmann::json& tempArray =
1966                             sensorsAsyncResp->asyncResp->res
1967                                 .jsonValue[fieldName];
1968                         if (fieldName == "PowerControl")
1969                         {
1970                             if (tempArray.empty())
1971                             {
1972                                 // Put multiple "sensors" into a single
1973                                 // PowerControl. Follows MemberId naming and
1974                                 // naming in power.hpp.
1975                                 nlohmann::json::object_t power;
1976                                 boost::urls::url url = boost::urls::format(
1977                                     "/redfish/v1/Chassis/{}/{}",
1978                                     sensorsAsyncResp->chassisId,
1979                                     sensorsAsyncResp->chassisSubNode);
1980                                 url.set_fragment(
1981                                     (""_json_pointer / fieldName / "0")
1982                                         .to_string());
1983                                 power["@odata.id"] = std::move(url);
1984                                 tempArray.emplace_back(std::move(power));
1985                             }
1986                             sensorJson = &(tempArray.back());
1987                         }
1988                         else if (fieldName == "PowerSupplies")
1989                         {
1990                             if (inventoryItem != nullptr)
1991                             {
1992                                 sensorJson = &(getPowerSupply(
1993                                     tempArray, *inventoryItem,
1994                                     sensorsAsyncResp->chassisId));
1995                             }
1996                         }
1997                         else if (fieldName == "Members")
1998                         {
1999                             std::string sensorId =
2000                                 redfish::sensor_utils::getSensorId(sensorName,
2001                                                                    sensorType);
2002 
2003                             nlohmann::json::object_t member;
2004                             member["@odata.id"] = boost::urls::format(
2005                                 "/redfish/v1/Chassis/{}/{}/{}",
2006                                 sensorsAsyncResp->chassisId,
2007                                 sensorsAsyncResp->chassisSubNode, sensorId);
2008                             tempArray.emplace_back(std::move(member));
2009                             sensorJson = &(tempArray.back());
2010                         }
2011                         else
2012                         {
2013                             nlohmann::json::object_t member;
2014                             boost::urls::url url = boost::urls::format(
2015                                 "/redfish/v1/Chassis/{}/{}",
2016                                 sensorsAsyncResp->chassisId,
2017                                 sensorsAsyncResp->chassisSubNode);
2018                             url.set_fragment(
2019                                 (""_json_pointer / fieldName).to_string());
2020                             member["@odata.id"] = std::move(url);
2021                             tempArray.emplace_back(std::move(member));
2022                             sensorJson = &(tempArray.back());
2023                         }
2024                     }
2025 
2026                     if (sensorJson != nullptr)
2027                     {
2028                         objectInterfacesToJson(
2029                             sensorName, sensorType, chassisSubNode,
2030                             objDictEntry.second, *sensorJson, inventoryItem);
2031 
2032                         std::string path = "/xyz/openbmc_project/sensors/";
2033                         path += sensorType;
2034                         path += "/";
2035                         path += sensorName;
2036                         sensorsAsyncResp->addMetadata(*sensorJson, path);
2037                     }
2038                 }
2039                 if (sensorsAsyncResp.use_count() == 1)
2040                 {
2041                     sortJSONResponse(sensorsAsyncResp);
2042                     if (chassisSubNode ==
2043                             sensor_utils::ChassisSubNode::sensorsNode &&
2044                         sensorsAsyncResp->efficientExpand)
2045                     {
2046                         sensorsAsyncResp->asyncResp->res
2047                             .jsonValue["Members@odata.count"] =
2048                             sensorsAsyncResp->asyncResp->res
2049                                 .jsonValue["Members"]
2050                                 .size();
2051                     }
2052                     else if (chassisSubNode ==
2053                              sensor_utils::ChassisSubNode::thermalNode)
2054                     {
2055                         populateFanRedundancy(sensorsAsyncResp);
2056                     }
2057                 }
2058                 BMCWEB_LOG_DEBUG("getManagedObjectsCb exit");
2059             });
2060     }
2061     BMCWEB_LOG_DEBUG("getSensorData exit");
2062 }
2063 
2064 inline void
2065     processSensorList(const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
2066                       const std::shared_ptr<std::set<std::string>>& sensorNames)
2067 {
2068     auto getConnectionCb = [sensorsAsyncResp, sensorNames](
2069                                const std::set<std::string>& connections) {
2070         BMCWEB_LOG_DEBUG("getConnectionCb enter");
2071         auto getInventoryItemsCb =
2072             [sensorsAsyncResp, sensorNames, connections](
2073                 const std::shared_ptr<std::vector<InventoryItem>>&
2074                     inventoryItems) mutable {
2075                 BMCWEB_LOG_DEBUG("getInventoryItemsCb enter");
2076                 // Get sensor data and store results in JSON
2077                 getSensorData(sensorsAsyncResp, sensorNames, connections,
2078                               inventoryItems);
2079                 BMCWEB_LOG_DEBUG("getInventoryItemsCb exit");
2080             };
2081 
2082         // Get inventory items associated with sensors
2083         getInventoryItems(sensorsAsyncResp, sensorNames,
2084                           std::move(getInventoryItemsCb));
2085 
2086         BMCWEB_LOG_DEBUG("getConnectionCb exit");
2087     };
2088 
2089     // Get set of connections that provide sensor values
2090     getConnections(sensorsAsyncResp, sensorNames, std::move(getConnectionCb));
2091 }
2092 
2093 /**
2094  * @brief Entry point for retrieving sensors data related to requested
2095  *        chassis.
2096  * @param SensorsAsyncResp   Pointer to object holding response data
2097  */
2098 inline void
2099     getChassisData(const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp)
2100 {
2101     BMCWEB_LOG_DEBUG("getChassisData enter");
2102     auto getChassisCb =
2103         [sensorsAsyncResp](
2104             const std::shared_ptr<std::set<std::string>>& sensorNames) {
2105             BMCWEB_LOG_DEBUG("getChassisCb enter");
2106             processSensorList(sensorsAsyncResp, sensorNames);
2107             BMCWEB_LOG_DEBUG("getChassisCb exit");
2108         };
2109     // SensorCollection doesn't contain the Redundancy property
2110     if (sensorsAsyncResp->chassisSubNode != sensors::sensorsNodeStr)
2111     {
2112         sensorsAsyncResp->asyncResp->res.jsonValue["Redundancy"] =
2113             nlohmann::json::array();
2114     }
2115     // Get set of sensors in chassis
2116     getChassis(sensorsAsyncResp->asyncResp, sensorsAsyncResp->chassisId,
2117                sensorsAsyncResp->chassisSubNode, sensorsAsyncResp->types,
2118                std::move(getChassisCb));
2119     BMCWEB_LOG_DEBUG("getChassisData exit");
2120 }
2121 
2122 /**
2123  * @brief Find the requested sensorName in the list of all sensors supplied by
2124  * the chassis node
2125  *
2126  * @param sensorName   The sensor name supplied in the PATCH request
2127  * @param sensorsList  The list of sensors managed by the chassis node
2128  * @param sensorsModified  The list of sensors that were found as a result of
2129  *                         repeated calls to this function
2130  */
2131 inline bool findSensorNameUsingSensorPath(
2132     std::string_view sensorName, const std::set<std::string>& sensorsList,
2133     std::set<std::string>& sensorsModified)
2134 {
2135     for (const auto& chassisSensor : sensorsList)
2136     {
2137         sdbusplus::message::object_path path(chassisSensor);
2138         std::string thisSensorName = path.filename();
2139         if (thisSensorName.empty())
2140         {
2141             continue;
2142         }
2143         if (thisSensorName == sensorName)
2144         {
2145             sensorsModified.emplace(chassisSensor);
2146             return true;
2147         }
2148     }
2149     return false;
2150 }
2151 
2152 /**
2153  * @brief Entry point for overriding sensor values of given sensor
2154  *
2155  * @param sensorAsyncResp   response object
2156  * @param allCollections   Collections extract from sensors' request patch info
2157  * @param chassisSubNode   Chassis Node for which the query has to happen
2158  */
2159 inline void setSensorsOverride(
2160     const std::shared_ptr<SensorsAsyncResp>& sensorAsyncResp,
2161     std::unordered_map<std::string, std::vector<nlohmann::json::object_t>>&
2162         allCollections)
2163 {
2164     BMCWEB_LOG_INFO("setSensorsOverride for subNode{}",
2165                     sensorAsyncResp->chassisSubNode);
2166 
2167     std::string_view propertyValueName;
2168     std::unordered_map<std::string, std::pair<double, std::string>> overrideMap;
2169     std::string memberId;
2170     double value = 0.0;
2171     for (auto& collectionItems : allCollections)
2172     {
2173         if (collectionItems.first == "Temperatures")
2174         {
2175             propertyValueName = "ReadingCelsius";
2176         }
2177         else if (collectionItems.first == "Fans")
2178         {
2179             propertyValueName = "Reading";
2180         }
2181         else
2182         {
2183             propertyValueName = "ReadingVolts";
2184         }
2185         for (auto& item : collectionItems.second)
2186         {
2187             if (!json_util::readJsonObject( //
2188                     item, sensorAsyncResp->asyncResp->res, //
2189                     "MemberId", memberId, //
2190                     propertyValueName, value //
2191                     ))
2192             {
2193                 return;
2194             }
2195             overrideMap.emplace(memberId,
2196                                 std::make_pair(value, collectionItems.first));
2197         }
2198     }
2199 
2200     auto getChassisSensorListCb = [sensorAsyncResp, overrideMap,
2201                                    propertyValueNameStr =
2202                                        std::string(propertyValueName)](
2203                                       const std::shared_ptr<
2204                                           std::set<std::string>>& sensorsList) {
2205         // Match sensor names in the PATCH request to those managed by the
2206         // chassis node
2207         const std::shared_ptr<std::set<std::string>> sensorNames =
2208             std::make_shared<std::set<std::string>>();
2209         for (const auto& item : overrideMap)
2210         {
2211             const auto& sensor = item.first;
2212             std::pair<std::string, std::string> sensorNameType =
2213                 redfish::sensor_utils::splitSensorNameAndType(sensor);
2214             if (!findSensorNameUsingSensorPath(sensorNameType.second,
2215                                                *sensorsList, *sensorNames))
2216             {
2217                 BMCWEB_LOG_INFO("Unable to find memberId {}", item.first);
2218                 messages::resourceNotFound(sensorAsyncResp->asyncResp->res,
2219                                            item.second.second, item.first);
2220                 return;
2221             }
2222         }
2223         // Get the connection to which the memberId belongs
2224         auto getObjectsWithConnectionCb = [sensorAsyncResp, overrideMap,
2225                                            propertyValueNameStr](
2226                                               const std::set<
2227                                                   std::string>& /*connections*/,
2228                                               const std::set<std::pair<
2229                                                   std::string, std::string>>&
2230                                                   objectsWithConnection) {
2231             if (objectsWithConnection.size() != overrideMap.size())
2232             {
2233                 BMCWEB_LOG_INFO(
2234                     "Unable to find all objects with proper connection {} requested {}",
2235                     objectsWithConnection.size(), overrideMap.size());
2236                 messages::resourceNotFound(
2237                     sensorAsyncResp->asyncResp->res,
2238                     sensorAsyncResp->chassisSubNode == sensors::thermalNodeStr
2239                         ? "Temperatures"
2240                         : "Voltages",
2241                     "Count");
2242                 return;
2243             }
2244             for (const auto& item : objectsWithConnection)
2245             {
2246                 sdbusplus::message::object_path path(item.first);
2247                 std::string sensorName = path.filename();
2248                 if (sensorName.empty())
2249                 {
2250                     messages::internalError(sensorAsyncResp->asyncResp->res);
2251                     return;
2252                 }
2253                 std::string id = redfish::sensor_utils::getSensorId(
2254                     sensorName, path.parent_path().filename());
2255 
2256                 const auto& iterator = overrideMap.find(id);
2257                 if (iterator == overrideMap.end())
2258                 {
2259                     BMCWEB_LOG_INFO("Unable to find sensor object{}",
2260                                     item.first);
2261                     messages::internalError(sensorAsyncResp->asyncResp->res);
2262                     return;
2263                 }
2264                 setDbusProperty(sensorAsyncResp->asyncResp,
2265                                 propertyValueNameStr, item.second, item.first,
2266                                 "xyz.openbmc_project.Sensor.Value", "Value",
2267                                 iterator->second.first);
2268             }
2269         };
2270         // Get object with connection for the given sensor name
2271         getObjectsWithConnection(sensorAsyncResp, sensorNames,
2272                                  std::move(getObjectsWithConnectionCb));
2273     };
2274     // get full sensor list for the given chassisId and cross verify the sensor.
2275     getChassis(sensorAsyncResp->asyncResp, sensorAsyncResp->chassisId,
2276                sensorAsyncResp->chassisSubNode, sensorAsyncResp->types,
2277                std::move(getChassisSensorListCb));
2278 }
2279 
2280 /**
2281  * @brief Retrieves mapping of Redfish URIs to sensor value property to D-Bus
2282  * path of the sensor.
2283  *
2284  * Function builds valid Redfish response for sensor query of given chassis and
2285  * node. It then builds metadata about Redfish<->D-Bus correlations and provides
2286  * it to caller in a callback.
2287  *
2288  * @param chassis   Chassis for which retrieval should be performed
2289  * @param node  Node (group) of sensors. See sensor_utils::node for supported
2290  * values
2291  * @param mapComplete   Callback to be called with retrieval result
2292  */
2293 template <typename Callback>
2294 inline void retrieveUriToDbusMap(
2295     const std::string& chassis, const std::string& node, Callback&& mapComplete)
2296 {
2297     decltype(sensors::paths)::const_iterator pathIt =
2298         std::find_if(sensors::paths.cbegin(), sensors::paths.cend(),
2299                      [&node](auto&& val) { return val.first == node; });
2300     if (pathIt == sensors::paths.cend())
2301     {
2302         BMCWEB_LOG_ERROR("Wrong node provided : {}", node);
2303         std::map<std::string, std::string> noop;
2304         mapComplete(boost::beast::http::status::bad_request, noop);
2305         return;
2306     }
2307 
2308     auto asyncResp = std::make_shared<bmcweb::AsyncResp>();
2309     auto callback =
2310         [asyncResp, mapCompleteCb = std::forward<Callback>(mapComplete)](
2311             const boost::beast::http::status status,
2312             const std::map<std::string, std::string>& uriToDbus) {
2313             mapCompleteCb(status, uriToDbus);
2314         };
2315 
2316     auto resp = std::make_shared<SensorsAsyncResp>(
2317         asyncResp, chassis, pathIt->second, node, std::move(callback));
2318     getChassisData(resp);
2319 }
2320 
2321 namespace sensors
2322 {
2323 
2324 inline void getChassisCallback(
2325     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2326     std::string_view chassisId, std::string_view chassisSubNode,
2327     const std::shared_ptr<std::set<std::string>>& sensorNames)
2328 {
2329     BMCWEB_LOG_DEBUG("getChassisCallback enter ");
2330 
2331     nlohmann::json& entriesArray = asyncResp->res.jsonValue["Members"];
2332     for (const std::string& sensor : *sensorNames)
2333     {
2334         BMCWEB_LOG_DEBUG("Adding sensor: {}", sensor);
2335 
2336         sdbusplus::message::object_path path(sensor);
2337         std::string sensorName = path.filename();
2338         if (sensorName.empty())
2339         {
2340             BMCWEB_LOG_ERROR("Invalid sensor path: {}", sensor);
2341             messages::internalError(asyncResp->res);
2342             return;
2343         }
2344         std::string type = path.parent_path().filename();
2345         std::string id = redfish::sensor_utils::getSensorId(sensorName, type);
2346 
2347         nlohmann::json::object_t member;
2348         member["@odata.id"] = boost::urls::format(
2349             "/redfish/v1/Chassis/{}/{}/{}", chassisId, chassisSubNode, id);
2350 
2351         entriesArray.emplace_back(std::move(member));
2352     }
2353 
2354     asyncResp->res.jsonValue["Members@odata.count"] = entriesArray.size();
2355     BMCWEB_LOG_DEBUG("getChassisCallback exit");
2356 }
2357 
2358 inline void handleSensorCollectionGet(
2359     App& app, const crow::Request& req,
2360     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2361     const std::string& chassisId)
2362 {
2363     query_param::QueryCapabilities capabilities = {
2364         .canDelegateExpandLevel = 1,
2365     };
2366     query_param::Query delegatedQuery;
2367     if (!redfish::setUpRedfishRouteWithDelegation(app, req, asyncResp,
2368                                                   delegatedQuery, capabilities))
2369     {
2370         return;
2371     }
2372 
2373     if (delegatedQuery.expandType != query_param::ExpandType::None)
2374     {
2375         // we perform efficient expand.
2376         auto sensorsAsyncResp = std::make_shared<SensorsAsyncResp>(
2377             asyncResp, chassisId, sensors::dbus::sensorPaths,
2378             sensors::sensorsNodeStr,
2379             /*efficientExpand=*/true);
2380         getChassisData(sensorsAsyncResp);
2381 
2382         BMCWEB_LOG_DEBUG(
2383             "SensorCollection doGet exit via efficient expand handler");
2384         return;
2385     }
2386 
2387     // We get all sensors as hyperlinkes in the chassis (this
2388     // implies we reply on the default query parameters handler)
2389     getChassis(asyncResp, chassisId, sensors::sensorsNodeStr, dbus::sensorPaths,
2390                std::bind_front(sensors::getChassisCallback, asyncResp,
2391                                chassisId, sensors::sensorsNodeStr));
2392 }
2393 
2394 inline void
2395     getSensorFromDbus(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2396                       const std::string& sensorPath,
2397                       const ::dbus::utility::MapperGetObject& mapperResponse)
2398 {
2399     if (mapperResponse.size() != 1)
2400     {
2401         messages::internalError(asyncResp->res);
2402         return;
2403     }
2404     const auto& valueIface = *mapperResponse.begin();
2405     const std::string& connectionName = valueIface.first;
2406     BMCWEB_LOG_DEBUG("Looking up {}", connectionName);
2407     BMCWEB_LOG_DEBUG("Path {}", sensorPath);
2408 
2409     ::dbus::utility::getAllProperties(
2410         *crow::connections::systemBus, connectionName, sensorPath, "",
2411         [asyncResp,
2412          sensorPath](const boost::system::error_code& ec,
2413                      const ::dbus::utility::DBusPropertiesMap& valuesDict) {
2414             if (ec)
2415             {
2416                 messages::internalError(asyncResp->res);
2417                 return;
2418             }
2419             sdbusplus::message::object_path path(sensorPath);
2420             std::string name = path.filename();
2421             path = path.parent_path();
2422             std::string type = path.filename();
2423             sensor_utils::objectPropertiesToJson(
2424                 name, type, sensor_utils::ChassisSubNode::sensorsNode,
2425                 valuesDict, asyncResp->res.jsonValue, nullptr);
2426         });
2427 }
2428 
2429 inline void handleSensorGet(App& app, const crow::Request& req,
2430                             const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2431                             const std::string& chassisId,
2432                             const std::string& sensorId)
2433 {
2434     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
2435     {
2436         return;
2437     }
2438     std::pair<std::string, std::string> nameType =
2439         redfish::sensor_utils::splitSensorNameAndType(sensorId);
2440     if (nameType.first.empty() || nameType.second.empty())
2441     {
2442         messages::resourceNotFound(asyncResp->res, sensorId, "Sensor");
2443         return;
2444     }
2445 
2446     asyncResp->res.jsonValue["@odata.id"] = boost::urls::format(
2447         "/redfish/v1/Chassis/{}/Sensors/{}", chassisId, sensorId);
2448 
2449     BMCWEB_LOG_DEBUG("Sensor doGet enter");
2450 
2451     constexpr std::array<std::string_view, 1> interfaces = {
2452         "xyz.openbmc_project.Sensor.Value"};
2453     std::string sensorPath = "/xyz/openbmc_project/sensors/" + nameType.first +
2454                              '/' + nameType.second;
2455     // Get a list of all of the sensors that implement Sensor.Value
2456     // and get the path and service name associated with the sensor
2457     ::dbus::utility::getDbusObject(
2458         sensorPath, interfaces,
2459         [asyncResp, sensorId,
2460          sensorPath](const boost::system::error_code& ec,
2461                      const ::dbus::utility::MapperGetObject& subtree) {
2462             BMCWEB_LOG_DEBUG("respHandler1 enter");
2463             if (ec == boost::system::errc::io_error)
2464             {
2465                 BMCWEB_LOG_WARNING("Sensor not found from getSensorPaths");
2466                 messages::resourceNotFound(asyncResp->res, sensorId, "Sensor");
2467                 return;
2468             }
2469             if (ec)
2470             {
2471                 messages::internalError(asyncResp->res);
2472                 BMCWEB_LOG_ERROR(
2473                     "Sensor getSensorPaths resp_handler: Dbus error {}", ec);
2474                 return;
2475             }
2476             getSensorFromDbus(asyncResp, sensorPath, subtree);
2477             BMCWEB_LOG_DEBUG("respHandler1 exit");
2478         });
2479 }
2480 
2481 } // namespace sensors
2482 
2483 inline void requestRoutesSensorCollection(App& app)
2484 {
2485     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Sensors/")
2486         .privileges(redfish::privileges::getSensorCollection)
2487         .methods(boost::beast::http::verb::get)(
2488             std::bind_front(sensors::handleSensorCollectionGet, std::ref(app)));
2489 }
2490 
2491 inline void requestRoutesSensor(App& app)
2492 {
2493     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Sensors/<str>/")
2494         .privileges(redfish::privileges::getSensor)
2495         .methods(boost::beast::http::verb::get)(
2496             std::bind_front(sensors::handleSensorGet, std::ref(app)));
2497 }
2498 
2499 } // namespace redfish
2500