xref: /openbmc/bmcweb/redfish-core/lib/sensors.hpp (revision 41868c66e03c9742d95a4428e686c0bf6dd82ab7)
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(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                         sdbusplus::asio::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     std::shared_ptr<SensorsAsyncResp> sensorsAsyncResp,
974     std::shared_ptr<std::vector<InventoryItem>> inventoryItems,
975     std::shared_ptr<std::set<std::string>> invConnections, Callback&& callback,
976     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     std::shared_ptr<SensorsAsyncResp> sensorsAsyncResp,
1307     std::shared_ptr<std::vector<InventoryItem>> inventoryItems,
1308     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         sdbusplus::asio::getProperty<std::string>(
1379             *crow::connections::systemBus, ledConnection, ledPath,
1380             "xyz.openbmc_project.Led.Physical", "State",
1381             std::move(respHandler));
1382     }
1383 
1384     BMCWEB_LOG_DEBUG("getInventoryLedData exit");
1385 }
1386 
1387 /**
1388  * @brief Gets LED data for LEDs associated with given inventory items.
1389  *
1390  * Gets the D-Bus connections (services) that provide LED data for the LEDs
1391  * associated with the specified inventory items.  Then gets the LED data from
1392  * each connection and stores it in the inventory item.
1393  *
1394  * This data is later used to provide sensor property values in the JSON
1395  * response.
1396  *
1397  * Finds the LED data asynchronously.  Invokes callback when information has
1398  * been obtained.
1399  *
1400  * The callback must have the following signature:
1401  *   @code
1402  *   callback()
1403  *   @endcode
1404  *
1405  * @param sensorsAsyncResp Pointer to object holding response data.
1406  * @param inventoryItems D-Bus inventory items associated with sensors.
1407  * @param callback Callback to invoke when inventory items have been obtained.
1408  */
1409 template <typename Callback>
1410 void getInventoryLeds(
1411     std::shared_ptr<SensorsAsyncResp> sensorsAsyncResp,
1412     std::shared_ptr<std::vector<InventoryItem>> inventoryItems,
1413     Callback&& callback)
1414 {
1415     BMCWEB_LOG_DEBUG("getInventoryLeds enter");
1416 
1417     const std::string path = "/xyz/openbmc_project";
1418     constexpr std::array<std::string_view, 1> interfaces = {
1419         "xyz.openbmc_project.Led.Physical"};
1420 
1421     // Make call to ObjectMapper to find all inventory items
1422     dbus::utility::getSubTree(
1423         path, 0, interfaces,
1424         [callback = std::forward<Callback>(callback), sensorsAsyncResp,
1425          inventoryItems](
1426             const boost::system::error_code& ec,
1427             const dbus::utility::MapperGetSubTreeResponse& subtree) mutable {
1428             // Response handler for parsing output from GetSubTree
1429             BMCWEB_LOG_DEBUG("getInventoryLeds respHandler enter");
1430             if (ec)
1431             {
1432                 messages::internalError(sensorsAsyncResp->asyncResp->res);
1433                 BMCWEB_LOG_ERROR("getInventoryLeds respHandler DBus error {}",
1434                                  ec);
1435                 return;
1436             }
1437 
1438             // Build map of LED object paths to connections
1439             std::shared_ptr<std::map<std::string, std::string>> ledConnections =
1440                 std::make_shared<std::map<std::string, std::string>>();
1441 
1442             // Loop through objects from GetSubTree
1443             for (const std::pair<std::string,
1444                                  std::vector<std::pair<
1445                                      std::string, std::vector<std::string>>>>&
1446                      object : subtree)
1447             {
1448                 // Check if object path is LED for one of the specified
1449                 // inventory items
1450                 const std::string& ledPath = object.first;
1451                 if (findInventoryItemForLed(*inventoryItems, ledPath) !=
1452                     nullptr)
1453                 {
1454                     // Add mapping from ledPath to connection
1455                     const std::string& connection =
1456                         object.second.begin()->first;
1457                     (*ledConnections)[ledPath] = connection;
1458                     BMCWEB_LOG_DEBUG("Added mapping {} -> {}", ledPath,
1459                                      connection);
1460                 }
1461             }
1462 
1463             getInventoryLedData(sensorsAsyncResp, inventoryItems,
1464                                 ledConnections, std::move(callback));
1465             BMCWEB_LOG_DEBUG("getInventoryLeds respHandler exit");
1466         });
1467     BMCWEB_LOG_DEBUG("getInventoryLeds exit");
1468 }
1469 
1470 /**
1471  * @brief Gets D-Bus data for Power Supply Attributes such as EfficiencyPercent
1472  *
1473  * Uses the specified connections (services) (currently assumes just one) to
1474  * obtain D-Bus data for Power Supply Attributes. Stores the resulting data in
1475  * the inventoryItems vector. Only stores data in Power Supply inventoryItems.
1476  *
1477  * This data is later used to provide sensor property values in the JSON
1478  * response.
1479  *
1480  * Finds the Power Supply Attributes data asynchronously.  Invokes callback
1481  * when data has been obtained.
1482  *
1483  * The callback must have the following signature:
1484  *   @code
1485  *   callback(std::shared_ptr<std::vector<InventoryItem>> inventoryItems)
1486  *   @endcode
1487  *
1488  * @param sensorsAsyncResp Pointer to object holding response data.
1489  * @param inventoryItems D-Bus inventory items associated with sensors.
1490  * @param psAttributesConnections Connections that provide data for the Power
1491  *        Supply Attributes
1492  * @param callback Callback to invoke when data has been obtained.
1493  */
1494 template <typename Callback>
1495 void getPowerSupplyAttributesData(
1496     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1497     std::shared_ptr<std::vector<InventoryItem>> inventoryItems,
1498     const std::map<std::string, std::string>& psAttributesConnections,
1499     Callback&& callback)
1500 {
1501     BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData enter");
1502 
1503     if (psAttributesConnections.empty())
1504     {
1505         BMCWEB_LOG_DEBUG("Can't find PowerSupplyAttributes, no connections!");
1506         callback(inventoryItems);
1507         return;
1508     }
1509 
1510     // Assuming just one connection (service) for now
1511     auto it = psAttributesConnections.begin();
1512 
1513     const std::string& psAttributesPath = (*it).first;
1514     const std::string& psAttributesConnection = (*it).second;
1515 
1516     // Response handler for Get DeratingFactor property
1517     auto respHandler = [sensorsAsyncResp, inventoryItems,
1518                         callback = std::forward<Callback>(callback)](
1519                            const boost::system::error_code& ec,
1520                            uint32_t value) mutable {
1521         BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData respHandler enter");
1522         if (ec)
1523         {
1524             BMCWEB_LOG_ERROR(
1525                 "getPowerSupplyAttributesData respHandler DBus error {}", ec);
1526             messages::internalError(sensorsAsyncResp->asyncResp->res);
1527             return;
1528         }
1529 
1530         BMCWEB_LOG_DEBUG("PS EfficiencyPercent value: {}", value);
1531         // Store value in Power Supply Inventory Items
1532         for (InventoryItem& inventoryItem : *inventoryItems)
1533         {
1534             if (inventoryItem.isPowerSupply)
1535             {
1536                 inventoryItem.powerSupplyEfficiencyPercent =
1537                     static_cast<int>(value);
1538             }
1539         }
1540 
1541         BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData respHandler exit");
1542         callback(inventoryItems);
1543     };
1544 
1545     // Get the DeratingFactor property for the PowerSupplyAttributes
1546     // Currently only property on the interface/only one we care about
1547     sdbusplus::asio::getProperty<uint32_t>(
1548         *crow::connections::systemBus, psAttributesConnection, psAttributesPath,
1549         "xyz.openbmc_project.Control.PowerSupplyAttributes", "DeratingFactor",
1550         std::move(respHandler));
1551 
1552     BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData exit");
1553 }
1554 
1555 /**
1556  * @brief Gets the Power Supply Attributes such as EfficiencyPercent
1557  *
1558  * Gets the D-Bus connection (service) that provides Power Supply Attributes
1559  * data. Then gets the Power Supply Attributes data from the connection
1560  * (currently just assumes 1 connection) and stores the data in the inventory
1561  * item.
1562  *
1563  * This data is later used to provide sensor property values in the JSON
1564  * response. DeratingFactor on D-Bus is mapped to EfficiencyPercent on Redfish.
1565  *
1566  * Finds the Power Supply Attributes data asynchronously. Invokes callback
1567  * when information has been obtained.
1568  *
1569  * The callback must have the following signature:
1570  *   @code
1571  *   callback(std::shared_ptr<std::vector<InventoryItem>> inventoryItems)
1572  *   @endcode
1573  *
1574  * @param sensorsAsyncResp Pointer to object holding response data.
1575  * @param inventoryItems D-Bus inventory items associated with sensors.
1576  * @param callback Callback to invoke when data has been obtained.
1577  */
1578 template <typename Callback>
1579 void getPowerSupplyAttributes(
1580     std::shared_ptr<SensorsAsyncResp> sensorsAsyncResp,
1581     std::shared_ptr<std::vector<InventoryItem>> inventoryItems,
1582     Callback&& callback)
1583 {
1584     BMCWEB_LOG_DEBUG("getPowerSupplyAttributes enter");
1585 
1586     // Only need the power supply attributes when the Power Schema
1587     if (sensorsAsyncResp->chassisSubNode != sensors::powerNodeStr)
1588     {
1589         BMCWEB_LOG_DEBUG("getPowerSupplyAttributes exit since not Power");
1590         callback(inventoryItems);
1591         return;
1592     }
1593 
1594     constexpr std::array<std::string_view, 1> interfaces = {
1595         "xyz.openbmc_project.Control.PowerSupplyAttributes"};
1596 
1597     // Make call to ObjectMapper to find the PowerSupplyAttributes service
1598     dbus::utility::getSubTree(
1599         "/xyz/openbmc_project", 0, interfaces,
1600         [callback = std::forward<Callback>(callback), sensorsAsyncResp,
1601          inventoryItems](
1602             const boost::system::error_code& ec,
1603             const dbus::utility::MapperGetSubTreeResponse& subtree) mutable {
1604             // Response handler for parsing output from GetSubTree
1605             BMCWEB_LOG_DEBUG("getPowerSupplyAttributes respHandler enter");
1606             if (ec)
1607             {
1608                 messages::internalError(sensorsAsyncResp->asyncResp->res);
1609                 BMCWEB_LOG_ERROR(
1610                     "getPowerSupplyAttributes respHandler DBus error {}", ec);
1611                 return;
1612             }
1613             if (subtree.empty())
1614             {
1615                 BMCWEB_LOG_DEBUG("Can't find Power Supply Attributes!");
1616                 callback(inventoryItems);
1617                 return;
1618             }
1619 
1620             // Currently we only support 1 power supply attribute, use this for
1621             // all the power supplies. Build map of object path to connection.
1622             // Assume just 1 connection and 1 path for now.
1623             std::map<std::string, std::string> psAttributesConnections;
1624 
1625             if (subtree[0].first.empty() || subtree[0].second.empty())
1626             {
1627                 BMCWEB_LOG_DEBUG("Power Supply Attributes mapper error!");
1628                 callback(inventoryItems);
1629                 return;
1630             }
1631 
1632             const std::string& psAttributesPath = subtree[0].first;
1633             const std::string& connection = subtree[0].second.begin()->first;
1634 
1635             if (connection.empty())
1636             {
1637                 BMCWEB_LOG_DEBUG("Power Supply Attributes mapper error!");
1638                 callback(inventoryItems);
1639                 return;
1640             }
1641 
1642             psAttributesConnections[psAttributesPath] = connection;
1643             BMCWEB_LOG_DEBUG("Added mapping {} -> {}", psAttributesPath,
1644                              connection);
1645 
1646             getPowerSupplyAttributesData(sensorsAsyncResp, inventoryItems,
1647                                          psAttributesConnections,
1648                                          std::move(callback));
1649             BMCWEB_LOG_DEBUG("getPowerSupplyAttributes respHandler exit");
1650         });
1651     BMCWEB_LOG_DEBUG("getPowerSupplyAttributes exit");
1652 }
1653 
1654 /**
1655  * @brief Gets inventory items associated with sensors.
1656  *
1657  * Finds the inventory items that are associated with the specified sensors.
1658  * Then gets D-Bus data for the inventory items, such as presence and VPD.
1659  *
1660  * This data is later used to provide sensor property values in the JSON
1661  * response.
1662  *
1663  * Finds the inventory items asynchronously.  Invokes callback when the
1664  * inventory items have been obtained.
1665  *
1666  * The callback must have the following signature:
1667  *   @code
1668  *   callback(std::shared_ptr<std::vector<InventoryItem>> inventoryItems)
1669  *   @endcode
1670  *
1671  * @param sensorsAsyncResp Pointer to object holding response data.
1672  * @param sensorNames All sensors within the current chassis.
1673  * implements ObjectManager.
1674  * @param callback Callback to invoke when inventory items have been obtained.
1675  */
1676 template <typename Callback>
1677 inline void
1678     getInventoryItems(std::shared_ptr<SensorsAsyncResp> sensorsAsyncResp,
1679                       const std::shared_ptr<std::set<std::string>> sensorNames,
1680                       Callback&& callback)
1681 {
1682     BMCWEB_LOG_DEBUG("getInventoryItems enter");
1683     auto getInventoryItemAssociationsCb =
1684         [sensorsAsyncResp, callback = std::forward<Callback>(callback)](
1685             std::shared_ptr<std::vector<InventoryItem>>
1686                 inventoryItems) mutable {
1687             BMCWEB_LOG_DEBUG("getInventoryItemAssociationsCb enter");
1688             auto getInventoryItemsConnectionsCb =
1689                 [sensorsAsyncResp, inventoryItems,
1690                  callback = std::forward<Callback>(callback)](
1691                     std::shared_ptr<std::set<std::string>>
1692                         invConnections) mutable {
1693                     BMCWEB_LOG_DEBUG("getInventoryItemsConnectionsCb enter");
1694                     auto getInventoryItemsDataCb =
1695                         [sensorsAsyncResp, inventoryItems,
1696                          callback =
1697                              std::forward<Callback>(callback)]() mutable {
1698                             BMCWEB_LOG_DEBUG("getInventoryItemsDataCb enter");
1699 
1700                             auto getInventoryLedsCb =
1701                                 [sensorsAsyncResp, inventoryItems,
1702                                  callback = std::forward<Callback>(
1703                                      callback)]() mutable {
1704                                     BMCWEB_LOG_DEBUG(
1705                                         "getInventoryLedsCb enter");
1706                                     // Find Power Supply Attributes and get the
1707                                     // data
1708                                     getPowerSupplyAttributes(
1709                                         sensorsAsyncResp, inventoryItems,
1710                                         std::move(callback));
1711                                     BMCWEB_LOG_DEBUG("getInventoryLedsCb exit");
1712                                 };
1713 
1714                             // Find led connections and get the data
1715                             getInventoryLeds(sensorsAsyncResp, inventoryItems,
1716                                              std::move(getInventoryLedsCb));
1717                             BMCWEB_LOG_DEBUG("getInventoryItemsDataCb exit");
1718                         };
1719 
1720                     // Get inventory item data from connections
1721                     getInventoryItemsData(sensorsAsyncResp, inventoryItems,
1722                                           invConnections,
1723                                           std::move(getInventoryItemsDataCb));
1724                     BMCWEB_LOG_DEBUG("getInventoryItemsConnectionsCb exit");
1725                 };
1726 
1727             // Get connections that provide inventory item data
1728             getInventoryItemsConnections(
1729                 sensorsAsyncResp, inventoryItems,
1730                 std::move(getInventoryItemsConnectionsCb));
1731             BMCWEB_LOG_DEBUG("getInventoryItemAssociationsCb exit");
1732         };
1733 
1734     // Get associations from sensors to inventory items
1735     getInventoryItemAssociations(sensorsAsyncResp, sensorNames,
1736                                  std::move(getInventoryItemAssociationsCb));
1737     BMCWEB_LOG_DEBUG("getInventoryItems exit");
1738 }
1739 
1740 /**
1741  * @brief Returns JSON PowerSupply object for the specified inventory item.
1742  *
1743  * Searches for a JSON PowerSupply object that matches the specified inventory
1744  * item.  If one is not found, a new PowerSupply object is added to the JSON
1745  * array.
1746  *
1747  * Multiple sensors are often associated with one power supply inventory item.
1748  * As a result, multiple sensor values are stored in one JSON PowerSupply
1749  * object.
1750  *
1751  * @param powerSupplyArray JSON array containing Redfish PowerSupply objects.
1752  * @param inventoryItem Inventory item for the power supply.
1753  * @param chassisId Chassis that contains the power supply.
1754  * @return JSON PowerSupply object for the specified inventory item.
1755  */
1756 inline nlohmann::json& getPowerSupply(nlohmann::json& powerSupplyArray,
1757                                       const InventoryItem& inventoryItem,
1758                                       const std::string& chassisId)
1759 {
1760     std::string nameS;
1761     nameS.resize(inventoryItem.name.size());
1762     std::ranges::replace_copy(inventoryItem.name, nameS.begin(), '_', ' ');
1763     // Check if matching PowerSupply object already exists in JSON array
1764     for (nlohmann::json& powerSupply : powerSupplyArray)
1765     {
1766         nlohmann::json::iterator nameIt = powerSupply.find("Name");
1767         if (nameIt == powerSupply.end())
1768         {
1769             continue;
1770         }
1771         const std::string* name = nameIt->get_ptr<std::string*>();
1772         if (name == nullptr)
1773         {
1774             continue;
1775         }
1776         if (nameS == *name)
1777         {
1778             return powerSupply;
1779         }
1780     }
1781 
1782     // Add new PowerSupply object to JSON array
1783     powerSupplyArray.push_back({});
1784     nlohmann::json& powerSupply = powerSupplyArray.back();
1785     boost::urls::url url =
1786         boost::urls::format("/redfish/v1/Chassis/{}/Power", chassisId);
1787     url.set_fragment(("/PowerSupplies"_json_pointer).to_string());
1788     powerSupply["@odata.id"] = std::move(url);
1789     std::string escaped;
1790     escaped.resize(inventoryItem.name.size());
1791     std::ranges::replace_copy(inventoryItem.name, escaped.begin(), '_', ' ');
1792     powerSupply["Name"] = std::move(escaped);
1793     powerSupply["Manufacturer"] = inventoryItem.manufacturer;
1794     powerSupply["Model"] = inventoryItem.model;
1795     powerSupply["PartNumber"] = inventoryItem.partNumber;
1796     powerSupply["SerialNumber"] = inventoryItem.serialNumber;
1797     sensor_utils::setLedState(powerSupply, &inventoryItem);
1798 
1799     if (inventoryItem.powerSupplyEfficiencyPercent >= 0)
1800     {
1801         powerSupply["EfficiencyPercent"] =
1802             inventoryItem.powerSupplyEfficiencyPercent;
1803     }
1804 
1805     powerSupply["Status"]["State"] =
1806         sensor_utils::getState(&inventoryItem, true);
1807     const char* health = inventoryItem.isFunctional ? "OK" : "Critical";
1808     powerSupply["Status"]["Health"] = health;
1809 
1810     return powerSupply;
1811 }
1812 
1813 /**
1814  * @brief Gets the values of the specified sensors.
1815  *
1816  * Stores the results as JSON in the SensorsAsyncResp.
1817  *
1818  * Gets the sensor values asynchronously.  Stores the results later when the
1819  * information has been obtained.
1820  *
1821  * The sensorNames set contains all requested sensors for the current chassis.
1822  *
1823  * To minimize the number of DBus calls, the DBus method
1824  * org.freedesktop.DBus.ObjectManager.GetManagedObjects() is used to get the
1825  * values of all sensors provided by a connection (service).
1826  *
1827  * The connections set contains all the connections that provide sensor values.
1828  *
1829  * The InventoryItem vector contains D-Bus inventory items associated with the
1830  * sensors.  Inventory item data is needed for some Redfish sensor properties.
1831  *
1832  * @param SensorsAsyncResp Pointer to object holding response data.
1833  * @param sensorNames All requested sensors within the current chassis.
1834  * @param connections Connections that provide sensor values.
1835  * implements ObjectManager.
1836  * @param inventoryItems Inventory items associated with the sensors.
1837  */
1838 inline void getSensorData(
1839     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1840     const std::shared_ptr<std::set<std::string>>& sensorNames,
1841     const std::set<std::string>& connections,
1842     const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems)
1843 {
1844     BMCWEB_LOG_DEBUG("getSensorData enter");
1845     // Get managed objects from all services exposing sensors
1846     for (const std::string& connection : connections)
1847     {
1848         sdbusplus::message::object_path sensorPath(
1849             "/xyz/openbmc_project/sensors");
1850         dbus::utility::getManagedObjects(
1851             connection, sensorPath,
1852             [sensorsAsyncResp, sensorNames,
1853              inventoryItems](const boost::system::error_code& ec,
1854                              const dbus::utility::ManagedObjectType& resp) {
1855                 BMCWEB_LOG_DEBUG("getManagedObjectsCb enter");
1856                 if (ec)
1857                 {
1858                     BMCWEB_LOG_ERROR("getManagedObjectsCb DBUS error: {}", ec);
1859                     messages::internalError(sensorsAsyncResp->asyncResp->res);
1860                     return;
1861                 }
1862                 auto chassisSubNode = sensor_utils::chassisSubNodeFromString(
1863                     sensorsAsyncResp->chassisSubNode);
1864                 // Go through all objects and update response with sensor data
1865                 for (const auto& objDictEntry : resp)
1866                 {
1867                     const std::string& objPath =
1868                         static_cast<const std::string&>(objDictEntry.first);
1869                     BMCWEB_LOG_DEBUG("getManagedObjectsCb parsing object {}",
1870                                      objPath);
1871 
1872                     std::vector<std::string> split;
1873                     // Reserve space for
1874                     // /xyz/openbmc_project/sensors/<name>/<subname>
1875                     split.reserve(6);
1876                     // NOLINTNEXTLINE
1877                     bmcweb::split(split, objPath, '/');
1878                     if (split.size() < 6)
1879                     {
1880                         BMCWEB_LOG_ERROR("Got path that isn't long enough {}",
1881                                          objPath);
1882                         continue;
1883                     }
1884                     // These indexes aren't intuitive, as split puts an empty
1885                     // string at the beginning
1886                     const std::string& sensorType = split[4];
1887                     const std::string& sensorName = split[5];
1888                     BMCWEB_LOG_DEBUG("sensorName {} sensorType {}", sensorName,
1889                                      sensorType);
1890                     if (sensorNames->find(objPath) == sensorNames->end())
1891                     {
1892                         BMCWEB_LOG_DEBUG("{} not in sensor list ", sensorName);
1893                         continue;
1894                     }
1895 
1896                     // Find inventory item (if any) associated with sensor
1897                     InventoryItem* inventoryItem =
1898                         findInventoryItemForSensor(inventoryItems, objPath);
1899 
1900                     const std::string& sensorSchema =
1901                         sensorsAsyncResp->chassisSubNode;
1902 
1903                     nlohmann::json* sensorJson = nullptr;
1904 
1905                     if (sensorSchema == sensors::sensorsNodeStr &&
1906                         !sensorsAsyncResp->efficientExpand)
1907                     {
1908                         std::string sensorId =
1909                             redfish::sensor_utils::getSensorId(sensorName,
1910                                                                sensorType);
1911 
1912                         sensorsAsyncResp->asyncResp->res
1913                             .jsonValue["@odata.id"] = boost::urls::format(
1914                             "/redfish/v1/Chassis/{}/{}/{}",
1915                             sensorsAsyncResp->chassisId,
1916                             sensorsAsyncResp->chassisSubNode, sensorId);
1917                         sensorJson =
1918                             &(sensorsAsyncResp->asyncResp->res.jsonValue);
1919                     }
1920                     else
1921                     {
1922                         std::string fieldName;
1923                         if (sensorsAsyncResp->efficientExpand)
1924                         {
1925                             fieldName = "Members";
1926                         }
1927                         else if (sensorType == "temperature")
1928                         {
1929                             fieldName = "Temperatures";
1930                         }
1931                         else if (sensorType == "fan" ||
1932                                  sensorType == "fan_tach" ||
1933                                  sensorType == "fan_pwm")
1934                         {
1935                             fieldName = "Fans";
1936                         }
1937                         else if (sensorType == "voltage")
1938                         {
1939                             fieldName = "Voltages";
1940                         }
1941                         else if (sensorType == "power")
1942                         {
1943                             if (sensorName == "total_power")
1944                             {
1945                                 fieldName = "PowerControl";
1946                             }
1947                             else if ((inventoryItem != nullptr) &&
1948                                      (inventoryItem->isPowerSupply))
1949                             {
1950                                 fieldName = "PowerSupplies";
1951                             }
1952                             else
1953                             {
1954                                 // Other power sensors are in SensorCollection
1955                                 continue;
1956                             }
1957                         }
1958                         else
1959                         {
1960                             BMCWEB_LOG_ERROR(
1961                                 "Unsure how to handle sensorType {}",
1962                                 sensorType);
1963                             continue;
1964                         }
1965 
1966                         nlohmann::json& tempArray =
1967                             sensorsAsyncResp->asyncResp->res
1968                                 .jsonValue[fieldName];
1969                         if (fieldName == "PowerControl")
1970                         {
1971                             if (tempArray.empty())
1972                             {
1973                                 // Put multiple "sensors" into a single
1974                                 // PowerControl. Follows MemberId naming and
1975                                 // naming in power.hpp.
1976                                 nlohmann::json::object_t power;
1977                                 boost::urls::url url = boost::urls::format(
1978                                     "/redfish/v1/Chassis/{}/{}",
1979                                     sensorsAsyncResp->chassisId,
1980                                     sensorsAsyncResp->chassisSubNode);
1981                                 url.set_fragment(
1982                                     (""_json_pointer / fieldName / "0")
1983                                         .to_string());
1984                                 power["@odata.id"] = std::move(url);
1985                                 tempArray.emplace_back(std::move(power));
1986                             }
1987                             sensorJson = &(tempArray.back());
1988                         }
1989                         else if (fieldName == "PowerSupplies")
1990                         {
1991                             if (inventoryItem != nullptr)
1992                             {
1993                                 sensorJson = &(getPowerSupply(
1994                                     tempArray, *inventoryItem,
1995                                     sensorsAsyncResp->chassisId));
1996                             }
1997                         }
1998                         else if (fieldName == "Members")
1999                         {
2000                             std::string sensorId =
2001                                 redfish::sensor_utils::getSensorId(sensorName,
2002                                                                    sensorType);
2003 
2004                             nlohmann::json::object_t member;
2005                             member["@odata.id"] = boost::urls::format(
2006                                 "/redfish/v1/Chassis/{}/{}/{}",
2007                                 sensorsAsyncResp->chassisId,
2008                                 sensorsAsyncResp->chassisSubNode, sensorId);
2009                             tempArray.emplace_back(std::move(member));
2010                             sensorJson = &(tempArray.back());
2011                         }
2012                         else
2013                         {
2014                             nlohmann::json::object_t member;
2015                             boost::urls::url url = boost::urls::format(
2016                                 "/redfish/v1/Chassis/{}/{}",
2017                                 sensorsAsyncResp->chassisId,
2018                                 sensorsAsyncResp->chassisSubNode);
2019                             url.set_fragment(
2020                                 (""_json_pointer / fieldName).to_string());
2021                             member["@odata.id"] = std::move(url);
2022                             tempArray.emplace_back(std::move(member));
2023                             sensorJson = &(tempArray.back());
2024                         }
2025                     }
2026 
2027                     if (sensorJson != nullptr)
2028                     {
2029                         objectInterfacesToJson(
2030                             sensorName, sensorType, chassisSubNode,
2031                             objDictEntry.second, *sensorJson, inventoryItem);
2032 
2033                         std::string path = "/xyz/openbmc_project/sensors/";
2034                         path += sensorType;
2035                         path += "/";
2036                         path += sensorName;
2037                         sensorsAsyncResp->addMetadata(*sensorJson, path);
2038                     }
2039                 }
2040                 if (sensorsAsyncResp.use_count() == 1)
2041                 {
2042                     sortJSONResponse(sensorsAsyncResp);
2043                     if (chassisSubNode ==
2044                             sensor_utils::ChassisSubNode::sensorsNode &&
2045                         sensorsAsyncResp->efficientExpand)
2046                     {
2047                         sensorsAsyncResp->asyncResp->res
2048                             .jsonValue["Members@odata.count"] =
2049                             sensorsAsyncResp->asyncResp->res
2050                                 .jsonValue["Members"]
2051                                 .size();
2052                     }
2053                     else if (chassisSubNode ==
2054                              sensor_utils::ChassisSubNode::thermalNode)
2055                     {
2056                         populateFanRedundancy(sensorsAsyncResp);
2057                     }
2058                 }
2059                 BMCWEB_LOG_DEBUG("getManagedObjectsCb exit");
2060             });
2061     }
2062     BMCWEB_LOG_DEBUG("getSensorData exit");
2063 }
2064 
2065 inline void
2066     processSensorList(const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
2067                       const std::shared_ptr<std::set<std::string>>& sensorNames)
2068 {
2069     auto getConnectionCb = [sensorsAsyncResp, sensorNames](
2070                                const std::set<std::string>& connections) {
2071         BMCWEB_LOG_DEBUG("getConnectionCb enter");
2072         auto getInventoryItemsCb =
2073             [sensorsAsyncResp, sensorNames, connections](
2074                 const std::shared_ptr<std::vector<InventoryItem>>&
2075                     inventoryItems) mutable {
2076                 BMCWEB_LOG_DEBUG("getInventoryItemsCb enter");
2077                 // Get sensor data and store results in JSON
2078                 getSensorData(sensorsAsyncResp, sensorNames, connections,
2079                               inventoryItems);
2080                 BMCWEB_LOG_DEBUG("getInventoryItemsCb exit");
2081             };
2082 
2083         // Get inventory items associated with sensors
2084         getInventoryItems(sensorsAsyncResp, sensorNames,
2085                           std::move(getInventoryItemsCb));
2086 
2087         BMCWEB_LOG_DEBUG("getConnectionCb exit");
2088     };
2089 
2090     // Get set of connections that provide sensor values
2091     getConnections(sensorsAsyncResp, sensorNames, std::move(getConnectionCb));
2092 }
2093 
2094 /**
2095  * @brief Entry point for retrieving sensors data related to requested
2096  *        chassis.
2097  * @param SensorsAsyncResp   Pointer to object holding response data
2098  */
2099 inline void
2100     getChassisData(const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp)
2101 {
2102     BMCWEB_LOG_DEBUG("getChassisData enter");
2103     auto getChassisCb =
2104         [sensorsAsyncResp](
2105             const std::shared_ptr<std::set<std::string>>& sensorNames) {
2106             BMCWEB_LOG_DEBUG("getChassisCb enter");
2107             processSensorList(sensorsAsyncResp, sensorNames);
2108             BMCWEB_LOG_DEBUG("getChassisCb exit");
2109         };
2110     // SensorCollection doesn't contain the Redundancy property
2111     if (sensorsAsyncResp->chassisSubNode != sensors::sensorsNodeStr)
2112     {
2113         sensorsAsyncResp->asyncResp->res.jsonValue["Redundancy"] =
2114             nlohmann::json::array();
2115     }
2116     // Get set of sensors in chassis
2117     getChassis(sensorsAsyncResp->asyncResp, sensorsAsyncResp->chassisId,
2118                sensorsAsyncResp->chassisSubNode, sensorsAsyncResp->types,
2119                std::move(getChassisCb));
2120     BMCWEB_LOG_DEBUG("getChassisData exit");
2121 }
2122 
2123 /**
2124  * @brief Find the requested sensorName in the list of all sensors supplied by
2125  * the chassis node
2126  *
2127  * @param sensorName   The sensor name supplied in the PATCH request
2128  * @param sensorsList  The list of sensors managed by the chassis node
2129  * @param sensorsModified  The list of sensors that were found as a result of
2130  *                         repeated calls to this function
2131  */
2132 inline bool findSensorNameUsingSensorPath(
2133     std::string_view sensorName, const std::set<std::string>& sensorsList,
2134     std::set<std::string>& sensorsModified)
2135 {
2136     for (const auto& chassisSensor : sensorsList)
2137     {
2138         sdbusplus::message::object_path path(chassisSensor);
2139         std::string thisSensorName = path.filename();
2140         if (thisSensorName.empty())
2141         {
2142             continue;
2143         }
2144         if (thisSensorName == sensorName)
2145         {
2146             sensorsModified.emplace(chassisSensor);
2147             return true;
2148         }
2149     }
2150     return false;
2151 }
2152 
2153 /**
2154  * @brief Entry point for overriding sensor values of given sensor
2155  *
2156  * @param sensorAsyncResp   response object
2157  * @param allCollections   Collections extract from sensors' request patch info
2158  * @param chassisSubNode   Chassis Node for which the query has to happen
2159  */
2160 inline void setSensorsOverride(
2161     const std::shared_ptr<SensorsAsyncResp>& sensorAsyncResp,
2162     std::unordered_map<std::string, std::vector<nlohmann::json::object_t>>&
2163         allCollections)
2164 {
2165     BMCWEB_LOG_INFO("setSensorsOverride for subNode{}",
2166                     sensorAsyncResp->chassisSubNode);
2167 
2168     std::string_view propertyValueName;
2169     std::unordered_map<std::string, std::pair<double, std::string>> overrideMap;
2170     std::string memberId;
2171     double value = 0.0;
2172     for (auto& collectionItems : allCollections)
2173     {
2174         if (collectionItems.first == "Temperatures")
2175         {
2176             propertyValueName = "ReadingCelsius";
2177         }
2178         else if (collectionItems.first == "Fans")
2179         {
2180             propertyValueName = "Reading";
2181         }
2182         else
2183         {
2184             propertyValueName = "ReadingVolts";
2185         }
2186         for (auto& item : collectionItems.second)
2187         {
2188             if (!json_util::readJsonObject(
2189                     item, sensorAsyncResp->asyncResp->res, "MemberId", memberId,
2190                     propertyValueName, value))
2191             {
2192                 return;
2193             }
2194             overrideMap.emplace(memberId,
2195                                 std::make_pair(value, collectionItems.first));
2196         }
2197     }
2198 
2199     auto getChassisSensorListCb = [sensorAsyncResp, overrideMap,
2200                                    propertyValueNameStr =
2201                                        std::string(propertyValueName)](
2202                                       const std::shared_ptr<
2203                                           std::set<std::string>>& sensorsList) {
2204         // Match sensor names in the PATCH request to those managed by the
2205         // chassis node
2206         const std::shared_ptr<std::set<std::string>> sensorNames =
2207             std::make_shared<std::set<std::string>>();
2208         for (const auto& item : overrideMap)
2209         {
2210             const auto& sensor = item.first;
2211             std::pair<std::string, std::string> sensorNameType =
2212                 redfish::sensor_utils::splitSensorNameAndType(sensor);
2213             if (!findSensorNameUsingSensorPath(sensorNameType.second,
2214                                                *sensorsList, *sensorNames))
2215             {
2216                 BMCWEB_LOG_INFO("Unable to find memberId {}", item.first);
2217                 messages::resourceNotFound(sensorAsyncResp->asyncResp->res,
2218                                            item.second.second, item.first);
2219                 return;
2220             }
2221         }
2222         // Get the connection to which the memberId belongs
2223         auto getObjectsWithConnectionCb = [sensorAsyncResp, overrideMap,
2224                                            propertyValueNameStr](
2225                                               const std::set<
2226                                                   std::string>& /*connections*/,
2227                                               const std::set<std::pair<
2228                                                   std::string, std::string>>&
2229                                                   objectsWithConnection) {
2230             if (objectsWithConnection.size() != overrideMap.size())
2231             {
2232                 BMCWEB_LOG_INFO(
2233                     "Unable to find all objects with proper connection {} requested {}",
2234                     objectsWithConnection.size(), overrideMap.size());
2235                 messages::resourceNotFound(
2236                     sensorAsyncResp->asyncResp->res,
2237                     sensorAsyncResp->chassisSubNode == sensors::thermalNodeStr
2238                         ? "Temperatures"
2239                         : "Voltages",
2240                     "Count");
2241                 return;
2242             }
2243             for (const auto& item : objectsWithConnection)
2244             {
2245                 sdbusplus::message::object_path path(item.first);
2246                 std::string sensorName = path.filename();
2247                 if (sensorName.empty())
2248                 {
2249                     messages::internalError(sensorAsyncResp->asyncResp->res);
2250                     return;
2251                 }
2252                 std::string id = redfish::sensor_utils::getSensorId(
2253                     sensorName, path.parent_path().filename());
2254 
2255                 const auto& iterator = overrideMap.find(id);
2256                 if (iterator == overrideMap.end())
2257                 {
2258                     BMCWEB_LOG_INFO("Unable to find sensor object{}",
2259                                     item.first);
2260                     messages::internalError(sensorAsyncResp->asyncResp->res);
2261                     return;
2262                 }
2263                 setDbusProperty(sensorAsyncResp->asyncResp,
2264                                 propertyValueNameStr, item.second, item.first,
2265                                 "xyz.openbmc_project.Sensor.Value", "Value",
2266                                 iterator->second.first);
2267             }
2268         };
2269         // Get object with connection for the given sensor name
2270         getObjectsWithConnection(sensorAsyncResp, sensorNames,
2271                                  std::move(getObjectsWithConnectionCb));
2272     };
2273     // get full sensor list for the given chassisId and cross verify the sensor.
2274     getChassis(sensorAsyncResp->asyncResp, sensorAsyncResp->chassisId,
2275                sensorAsyncResp->chassisSubNode, sensorAsyncResp->types,
2276                std::move(getChassisSensorListCb));
2277 }
2278 
2279 /**
2280  * @brief Retrieves mapping of Redfish URIs to sensor value property to D-Bus
2281  * path of the sensor.
2282  *
2283  * Function builds valid Redfish response for sensor query of given chassis and
2284  * node. It then builds metadata about Redfish<->D-Bus correlations and provides
2285  * it to caller in a callback.
2286  *
2287  * @param chassis   Chassis for which retrieval should be performed
2288  * @param node  Node (group) of sensors. See sensor_utils::node for supported
2289  * values
2290  * @param mapComplete   Callback to be called with retrieval result
2291  */
2292 template <typename Callback>
2293 inline void retrieveUriToDbusMap(
2294     const std::string& chassis, const std::string& node, Callback&& mapComplete)
2295 {
2296     decltype(sensors::paths)::const_iterator pathIt =
2297         std::find_if(sensors::paths.cbegin(), sensors::paths.cend(),
2298                      [&node](auto&& val) { return val.first == node; });
2299     if (pathIt == sensors::paths.cend())
2300     {
2301         BMCWEB_LOG_ERROR("Wrong node provided : {}", node);
2302         std::map<std::string, std::string> noop;
2303         mapComplete(boost::beast::http::status::bad_request, noop);
2304         return;
2305     }
2306 
2307     auto asyncResp = std::make_shared<bmcweb::AsyncResp>();
2308     auto callback =
2309         [asyncResp, mapCompleteCb = std::forward<Callback>(mapComplete)](
2310             const boost::beast::http::status status,
2311             const std::map<std::string, std::string>& uriToDbus) {
2312             mapCompleteCb(status, uriToDbus);
2313         };
2314 
2315     auto resp = std::make_shared<SensorsAsyncResp>(
2316         asyncResp, chassis, pathIt->second, node, std::move(callback));
2317     getChassisData(resp);
2318 }
2319 
2320 namespace sensors
2321 {
2322 
2323 inline void getChassisCallback(
2324     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2325     std::string_view chassisId, std::string_view chassisSubNode,
2326     const std::shared_ptr<std::set<std::string>>& sensorNames)
2327 {
2328     BMCWEB_LOG_DEBUG("getChassisCallback enter ");
2329 
2330     nlohmann::json& entriesArray = asyncResp->res.jsonValue["Members"];
2331     for (const std::string& sensor : *sensorNames)
2332     {
2333         BMCWEB_LOG_DEBUG("Adding sensor: {}", sensor);
2334 
2335         sdbusplus::message::object_path path(sensor);
2336         std::string sensorName = path.filename();
2337         if (sensorName.empty())
2338         {
2339             BMCWEB_LOG_ERROR("Invalid sensor path: {}", sensor);
2340             messages::internalError(asyncResp->res);
2341             return;
2342         }
2343         std::string type = path.parent_path().filename();
2344         std::string id = redfish::sensor_utils::getSensorId(sensorName, type);
2345 
2346         nlohmann::json::object_t member;
2347         member["@odata.id"] = boost::urls::format(
2348             "/redfish/v1/Chassis/{}/{}/{}", chassisId, chassisSubNode, id);
2349 
2350         entriesArray.emplace_back(std::move(member));
2351     }
2352 
2353     asyncResp->res.jsonValue["Members@odata.count"] = entriesArray.size();
2354     BMCWEB_LOG_DEBUG("getChassisCallback exit");
2355 }
2356 
2357 inline void handleSensorCollectionGet(
2358     App& app, const crow::Request& req,
2359     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2360     const std::string& chassisId)
2361 {
2362     query_param::QueryCapabilities capabilities = {
2363         .canDelegateExpandLevel = 1,
2364     };
2365     query_param::Query delegatedQuery;
2366     if (!redfish::setUpRedfishRouteWithDelegation(app, req, asyncResp,
2367                                                   delegatedQuery, capabilities))
2368     {
2369         return;
2370     }
2371 
2372     if (delegatedQuery.expandType != query_param::ExpandType::None)
2373     {
2374         // we perform efficient expand.
2375         auto sensorsAsyncResp = std::make_shared<SensorsAsyncResp>(
2376             asyncResp, chassisId, sensors::dbus::sensorPaths,
2377             sensors::sensorsNodeStr,
2378             /*efficientExpand=*/true);
2379         getChassisData(sensorsAsyncResp);
2380 
2381         BMCWEB_LOG_DEBUG(
2382             "SensorCollection doGet exit via efficient expand handler");
2383         return;
2384     }
2385 
2386     // We get all sensors as hyperlinkes in the chassis (this
2387     // implies we reply on the default query parameters handler)
2388     getChassis(asyncResp, chassisId, sensors::sensorsNodeStr, dbus::sensorPaths,
2389                std::bind_front(sensors::getChassisCallback, asyncResp,
2390                                chassisId, sensors::sensorsNodeStr));
2391 }
2392 
2393 inline void
2394     getSensorFromDbus(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2395                       const std::string& sensorPath,
2396                       const ::dbus::utility::MapperGetObject& mapperResponse)
2397 {
2398     if (mapperResponse.size() != 1)
2399     {
2400         messages::internalError(asyncResp->res);
2401         return;
2402     }
2403     const auto& valueIface = *mapperResponse.begin();
2404     const std::string& connectionName = valueIface.first;
2405     BMCWEB_LOG_DEBUG("Looking up {}", connectionName);
2406     BMCWEB_LOG_DEBUG("Path {}", sensorPath);
2407 
2408     sdbusplus::asio::getAllProperties(
2409         *crow::connections::systemBus, connectionName, sensorPath, "",
2410         [asyncResp,
2411          sensorPath](const boost::system::error_code& ec,
2412                      const ::dbus::utility::DBusPropertiesMap& valuesDict) {
2413             if (ec)
2414             {
2415                 messages::internalError(asyncResp->res);
2416                 return;
2417             }
2418             sdbusplus::message::object_path path(sensorPath);
2419             std::string name = path.filename();
2420             path = path.parent_path();
2421             std::string type = path.filename();
2422             sensor_utils::objectPropertiesToJson(
2423                 name, type, sensor_utils::ChassisSubNode::sensorsNode,
2424                 valuesDict, asyncResp->res.jsonValue, nullptr);
2425         });
2426 }
2427 
2428 inline void handleSensorGet(App& app, const crow::Request& req,
2429                             const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2430                             const std::string& chassisId,
2431                             const std::string& sensorId)
2432 {
2433     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
2434     {
2435         return;
2436     }
2437     std::pair<std::string, std::string> nameType =
2438         redfish::sensor_utils::splitSensorNameAndType(sensorId);
2439     if (nameType.first.empty() || nameType.second.empty())
2440     {
2441         messages::resourceNotFound(asyncResp->res, sensorId, "Sensor");
2442         return;
2443     }
2444 
2445     asyncResp->res.jsonValue["@odata.id"] = boost::urls::format(
2446         "/redfish/v1/Chassis/{}/Sensors/{}", chassisId, sensorId);
2447 
2448     BMCWEB_LOG_DEBUG("Sensor doGet enter");
2449 
2450     constexpr std::array<std::string_view, 1> interfaces = {
2451         "xyz.openbmc_project.Sensor.Value"};
2452     std::string sensorPath = "/xyz/openbmc_project/sensors/" + nameType.first +
2453                              '/' + nameType.second;
2454     // Get a list of all of the sensors that implement Sensor.Value
2455     // and get the path and service name associated with the sensor
2456     ::dbus::utility::getDbusObject(
2457         sensorPath, interfaces,
2458         [asyncResp, sensorId,
2459          sensorPath](const boost::system::error_code& ec,
2460                      const ::dbus::utility::MapperGetObject& subtree) {
2461             BMCWEB_LOG_DEBUG("respHandler1 enter");
2462             if (ec == boost::system::errc::io_error)
2463             {
2464                 BMCWEB_LOG_WARNING("Sensor not found from getSensorPaths");
2465                 messages::resourceNotFound(asyncResp->res, sensorId, "Sensor");
2466                 return;
2467             }
2468             if (ec)
2469             {
2470                 messages::internalError(asyncResp->res);
2471                 BMCWEB_LOG_ERROR(
2472                     "Sensor getSensorPaths resp_handler: Dbus error {}", ec);
2473                 return;
2474             }
2475             getSensorFromDbus(asyncResp, sensorPath, subtree);
2476             BMCWEB_LOG_DEBUG("respHandler1 exit");
2477         });
2478 }
2479 
2480 } // namespace sensors
2481 
2482 inline void requestRoutesSensorCollection(App& app)
2483 {
2484     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Sensors/")
2485         .privileges(redfish::privileges::getSensorCollection)
2486         .methods(boost::beast::http::verb::get)(
2487             std::bind_front(sensors::handleSensorCollectionGet, std::ref(app)));
2488 }
2489 
2490 inline void requestRoutesSensor(App& app)
2491 {
2492     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Sensors/<str>/")
2493         .privileges(redfish::privileges::getSensor)
2494         .methods(boost::beast::http::verb::get)(
2495             std::bind_front(sensors::handleSensorGet, std::ref(app)));
2496 }
2497 
2498 } // namespace redfish
2499