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
getSensorPaths()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
SensorsAsyncResp(const std::shared_ptr<bmcweb::AsyncResp> & asyncRespIn,const std::string & chassisIdIn,std::span<const std::string_view> typesIn,std::string_view subNode)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
SensorsAsyncResp(const std::shared_ptr<bmcweb::AsyncResp> & asyncRespIn,const std::string & chassisIdIn,std::span<const std::string_view> typesIn,std::string_view subNode,DataCompleteCb && creationComplete)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
SensorsAsyncResp(const std::shared_ptr<bmcweb::AsyncResp> & asyncRespIn,const std::string & chassisIdIn,std::span<const std::string_view> typesIn,const std::string_view & subNode,bool efficientExpandIn)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
~SensorsAsyncResp()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
addMetadata(const nlohmann::json & sensorObject,const std::string & dbusPath)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
updateUri(const std::string & name,const std::string & uri)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>
getObjectsWithConnection(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::set<std::string>> & sensorNames,Callback && 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>
getConnections(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::set<std::string>> & sensorNames,Callback && callback)306 void getConnections(const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
307 const std::shared_ptr<std::set<std::string>>& sensorNames,
308 Callback&& callback)
309 {
310 auto objectsWithConnectionCb =
311 [callback = std::forward<Callback>(callback)](
312 const std::set<std::string>& connections,
313 const std::set<std::pair<std::string, std::string>>&
314 /*objectsWithConnection*/) { callback(connections); };
315 getObjectsWithConnection(sensorsAsyncResp, sensorNames,
316 std::move(objectsWithConnectionCb));
317 }
318
319 /**
320 * @brief Shrinks the list of sensors for processing
321 * @param SensorsAysncResp The class holding the Redfish response
322 * @param allSensors A list of all the sensors associated to the
323 * chassis element (i.e. baseboard, front panel, etc...)
324 * @param activeSensors A list that is a reduction of the incoming
325 * allSensors list. Eliminate Thermal sensors when a Power request is
326 * made, and eliminate Power sensors when a Thermal request is made.
327 */
reduceSensorList(crow::Response & res,std::string_view chassisSubNode,std::span<const std::string_view> sensorTypes,const std::vector<std::string> * allSensors,const std::shared_ptr<std::set<std::string>> & activeSensors)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 * */
populateChassisNode(nlohmann::json & jsonValue,std::string_view chassisSubNode)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>
getChassis(const std::shared_ptr<bmcweb::AsyncResp> & asyncResp,std::string_view chassisId,std::string_view chassisSubNode,std::span<const std::string_view> sensorTypes,Callback && 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 */
objectInterfacesToJson(const std::string & sensorName,const std::string & sensorType,const sensor_utils::ChassisSubNode chassisSubNode,const dbus::utility::DBusInterfacesMap & interfacesDict,nlohmann::json & sensorJson,InventoryItem * inventoryItem)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
populateFanRedundancy(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp)510 inline void populateFanRedundancy(
511 const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp)
512 {
513 constexpr std::array<std::string_view, 1> interfaces = {
514 "xyz.openbmc_project.Control.FanRedundancy"};
515 dbus::utility::getSubTree(
516 "/xyz/openbmc_project/control", 2, interfaces,
517 [sensorsAsyncResp](
518 const boost::system::error_code& ec,
519 const dbus::utility::MapperGetSubTreeResponse& resp) {
520 if (ec)
521 {
522 return; // don't have to have this interface
523 }
524 for (const std::pair<std::string, dbus::utility::MapperServiceMap>&
525 pathPair : resp)
526 {
527 const std::string& path = pathPair.first;
528 const dbus::utility::MapperServiceMap& objDict =
529 pathPair.second;
530 if (objDict.empty())
531 {
532 continue; // this should be impossible
533 }
534
535 const std::string& owner = objDict.begin()->first;
536 dbus::utility::getAssociationEndPoints(
537 path + "/chassis",
538 [path, owner, sensorsAsyncResp](
539 const boost::system::error_code& ec2,
540 const dbus::utility::MapperEndPoints& endpoints) {
541 if (ec2)
542 {
543 return; // if they don't have an association we
544 // can't tell what chassis is
545 }
546 auto found = std::ranges::find_if(
547 endpoints,
548 [sensorsAsyncResp](const std::string& entry) {
549 return entry.find(
550 sensorsAsyncResp->chassisId) !=
551 std::string::npos;
552 });
553
554 if (found == endpoints.end())
555 {
556 return;
557 }
558 dbus::utility::getAllProperties(
559 *crow::connections::systemBus, owner, path,
560 "xyz.openbmc_project.Control.FanRedundancy",
561 [path, sensorsAsyncResp](
562 const boost::system::error_code& ec3,
563 const dbus::utility::DBusPropertiesMap& ret) {
564 if (ec3)
565 {
566 return; // don't have to have this
567 // interface
568 }
569
570 const uint8_t* allowedFailures = nullptr;
571 const std::vector<std::string>* collection =
572 nullptr;
573 const std::string* status = nullptr;
574
575 const bool success =
576 sdbusplus::unpackPropertiesNoThrow(
577 dbus_utils::UnpackErrorPrinter(), ret,
578 "AllowedFailures", allowedFailures,
579 "Collection", collection, "Status",
580 status);
581
582 if (!success)
583 {
584 messages::internalError(
585 sensorsAsyncResp->asyncResp->res);
586 return;
587 }
588
589 if (allowedFailures == nullptr ||
590 collection == nullptr || status == nullptr)
591 {
592 BMCWEB_LOG_ERROR(
593 "Invalid redundancy interface");
594 messages::internalError(
595 sensorsAsyncResp->asyncResp->res);
596 return;
597 }
598
599 sdbusplus::message::object_path objectPath(
600 path);
601 std::string name = objectPath.filename();
602 if (name.empty())
603 {
604 // this should be impossible
605 messages::internalError(
606 sensorsAsyncResp->asyncResp->res);
607 return;
608 }
609 std::ranges::replace(name, '_', ' ');
610
611 std::string health;
612
613 if (status->ends_with("Full"))
614 {
615 health = "OK";
616 }
617 else if (status->ends_with("Degraded"))
618 {
619 health = "Warning";
620 }
621 else
622 {
623 health = "Critical";
624 }
625 nlohmann::json::array_t redfishCollection;
626 const auto& fanRedfish =
627 sensorsAsyncResp->asyncResp->res
628 .jsonValue["Fans"];
629 for (const std::string& item : *collection)
630 {
631 sdbusplus::message::object_path itemPath(
632 item);
633 std::string itemName = itemPath.filename();
634 if (itemName.empty())
635 {
636 continue;
637 }
638 /*
639 todo(ed): merge patch that fixes the names
640 std::replace(itemName.begin(),
641 itemName.end(), '_', ' ');*/
642 auto schemaItem = std::ranges::find_if(
643 fanRedfish,
644 [itemName](const nlohmann::json& fan) {
645 return fan["Name"] == itemName;
646 });
647 if (schemaItem != fanRedfish.end())
648 {
649 nlohmann::json::object_t collectionId;
650 collectionId["@odata.id"] =
651 (*schemaItem)["@odata.id"];
652 redfishCollection.emplace_back(
653 std::move(collectionId));
654 }
655 else
656 {
657 BMCWEB_LOG_ERROR(
658 "failed to find fan in schema");
659 messages::internalError(
660 sensorsAsyncResp->asyncResp->res);
661 return;
662 }
663 }
664
665 size_t minNumNeeded =
666 collection->empty()
667 ? 0
668 : collection->size() - *allowedFailures;
669 nlohmann::json& jResp =
670 sensorsAsyncResp->asyncResp->res
671 .jsonValue["Redundancy"];
672
673 nlohmann::json::object_t redundancy;
674 boost::urls::url url = boost::urls::format(
675 "/redfish/v1/Chassis/{}/{}",
676 sensorsAsyncResp->chassisId,
677 sensorsAsyncResp->chassisSubNode);
678 url.set_fragment(
679 ("/Redundancy"_json_pointer / jResp.size())
680 .to_string());
681 redundancy["@odata.id"] = std::move(url);
682 redundancy["@odata.type"] =
683 "#Redundancy.v1_3_2.Redundancy";
684 redundancy["MinNumNeeded"] = minNumNeeded;
685 redundancy["Mode"] =
686 redundancy::RedundancyType::NPlusM;
687 redundancy["Name"] = name;
688 redundancy["RedundancySet"] = redfishCollection;
689 redundancy["Status"]["Health"] = health;
690 redundancy["Status"]["State"] =
691 resource::State::Enabled;
692
693 jResp.emplace_back(std::move(redundancy));
694 });
695 });
696 }
697 });
698 }
699
700 inline void
sortJSONResponse(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp)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 */
findInventoryItem(const std::shared_ptr<std::vector<InventoryItem>> & inventoryItems,const std::string & invItemObjPath)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 */
findInventoryItemForSensor(const std::shared_ptr<std::vector<InventoryItem>> & inventoryItems,const std::string & sensorObjPath)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 */
findInventoryItemForLed(std::vector<InventoryItem> & inventoryItems,const std::string & ledObjPath)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 */
addInventoryItem(const std::shared_ptr<std::vector<InventoryItem>> & inventoryItems,const std::string & invItemObjPath,const std::string & sensorObjPath)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 */
storeInventoryItemData(InventoryItem & inventoryItem,const dbus::utility::DBusInterfacesMap & interfacesDict)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>
getInventoryItemsData(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::vector<InventoryItem>> & inventoryItems,const std::shared_ptr<std::set<std::string>> & invConnections,Callback && callback,size_t invConnectionsIndex=0)972 void getInventoryItemsData(
973 const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
974 const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
975 const std::shared_ptr<std::set<std::string>>& invConnections,
976 Callback&& callback, size_t invConnectionsIndex = 0)
977 {
978 BMCWEB_LOG_DEBUG("getInventoryItemsData enter");
979
980 // If no more connections left, call callback
981 if (invConnectionsIndex >= invConnections->size())
982 {
983 callback();
984 BMCWEB_LOG_DEBUG("getInventoryItemsData exit");
985 return;
986 }
987
988 // Get inventory item data from current connection
989 auto it = invConnections->begin();
990 std::advance(it, invConnectionsIndex);
991 if (it != invConnections->end())
992 {
993 const std::string& invConnection = *it;
994
995 // Get all object paths and their interfaces for current connection
996 sdbusplus::message::object_path path("/xyz/openbmc_project/inventory");
997 dbus::utility::getManagedObjects(
998 invConnection, path,
999 [sensorsAsyncResp, inventoryItems, invConnections,
1000 callback = std::forward<Callback>(callback), invConnectionsIndex](
1001 const boost::system::error_code& ec,
1002 const dbus::utility::ManagedObjectType& resp) mutable {
1003 BMCWEB_LOG_DEBUG("getInventoryItemsData respHandler enter");
1004 if (ec)
1005 {
1006 BMCWEB_LOG_ERROR(
1007 "getInventoryItemsData respHandler DBus error {}", ec);
1008 messages::internalError(sensorsAsyncResp->asyncResp->res);
1009 return;
1010 }
1011
1012 // Loop through returned object paths
1013 for (const auto& objDictEntry : resp)
1014 {
1015 const std::string& objPath =
1016 static_cast<const std::string&>(objDictEntry.first);
1017
1018 // If this object path is one of the specified inventory
1019 // items
1020 InventoryItem* inventoryItem =
1021 findInventoryItem(inventoryItems, objPath);
1022 if (inventoryItem != nullptr)
1023 {
1024 // Store inventory data in InventoryItem
1025 storeInventoryItemData(*inventoryItem,
1026 objDictEntry.second);
1027 }
1028 }
1029
1030 // Recurse to get inventory item data from next connection
1031 getInventoryItemsData(sensorsAsyncResp, inventoryItems,
1032 invConnections, std::move(callback),
1033 invConnectionsIndex + 1);
1034
1035 BMCWEB_LOG_DEBUG("getInventoryItemsData respHandler exit");
1036 });
1037 }
1038
1039 BMCWEB_LOG_DEBUG("getInventoryItemsData exit");
1040 }
1041
1042 /**
1043 * @brief Gets connections that provide D-Bus data for inventory items.
1044 *
1045 * Gets the D-Bus connections (services) that provide data for the inventory
1046 * items that are associated with sensors.
1047 *
1048 * Finds the connections asynchronously. Invokes callback when information has
1049 * been obtained.
1050 *
1051 * The callback must have the following signature:
1052 * @code
1053 * callback(std::shared_ptr<std::set<std::string>> invConnections)
1054 * @endcode
1055 *
1056 * @param sensorsAsyncResp Pointer to object holding response data.
1057 * @param inventoryItems D-Bus inventory items associated with sensors.
1058 * @param callback Callback to invoke when connections have been obtained.
1059 */
1060 template <typename Callback>
getInventoryItemsConnections(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::vector<InventoryItem>> & inventoryItems,Callback && 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>
getInventoryItemAssociations(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::set<std::string>> & sensorNames,Callback && 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>
getInventoryLedData(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::vector<InventoryItem>> & inventoryItems,const std::shared_ptr<std::map<std::string,std::string>> & ledConnections,Callback && callback,size_t ledConnectionsIndex=0)1305 void getInventoryLedData(
1306 const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1307 const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
1308 const std::shared_ptr<std::map<std::string, std::string>>& ledConnections,
1309 Callback&& callback, size_t ledConnectionsIndex = 0)
1310 {
1311 BMCWEB_LOG_DEBUG("getInventoryLedData enter");
1312
1313 // If no more connections left, call callback
1314 if (ledConnectionsIndex >= ledConnections->size())
1315 {
1316 callback();
1317 BMCWEB_LOG_DEBUG("getInventoryLedData exit");
1318 return;
1319 }
1320
1321 // Get inventory item data from current connection
1322 auto it = ledConnections->begin();
1323 std::advance(it, ledConnectionsIndex);
1324 if (it != ledConnections->end())
1325 {
1326 const std::string& ledPath = (*it).first;
1327 const std::string& ledConnection = (*it).second;
1328 // Response handler for Get State property
1329 auto respHandler =
1330 [sensorsAsyncResp, inventoryItems, ledConnections, ledPath,
1331 callback = std::forward<Callback>(callback),
1332 ledConnectionsIndex](const boost::system::error_code& ec,
1333 const std::string& state) mutable {
1334 BMCWEB_LOG_DEBUG("getInventoryLedData respHandler enter");
1335 if (ec)
1336 {
1337 BMCWEB_LOG_ERROR(
1338 "getInventoryLedData respHandler DBus error {}", ec);
1339 messages::internalError(sensorsAsyncResp->asyncResp->res);
1340 return;
1341 }
1342
1343 BMCWEB_LOG_DEBUG("Led state: {}", state);
1344 // Find inventory item with this LED object path
1345 InventoryItem* inventoryItem =
1346 findInventoryItemForLed(*inventoryItems, ledPath);
1347 if (inventoryItem != nullptr)
1348 {
1349 // Store LED state in InventoryItem
1350 if (state.ends_with("On"))
1351 {
1352 inventoryItem->ledState = sensor_utils::LedState::ON;
1353 }
1354 else if (state.ends_with("Blink"))
1355 {
1356 inventoryItem->ledState = sensor_utils::LedState::BLINK;
1357 }
1358 else if (state.ends_with("Off"))
1359 {
1360 inventoryItem->ledState = sensor_utils::LedState::OFF;
1361 }
1362 else
1363 {
1364 inventoryItem->ledState =
1365 sensor_utils::LedState::UNKNOWN;
1366 }
1367 }
1368
1369 // Recurse to get LED data from next connection
1370 getInventoryLedData(sensorsAsyncResp, inventoryItems,
1371 ledConnections, std::move(callback),
1372 ledConnectionsIndex + 1);
1373
1374 BMCWEB_LOG_DEBUG("getInventoryLedData respHandler exit");
1375 };
1376
1377 // Get the State property for the current LED
1378 dbus::utility::getProperty<std::string>(
1379 ledConnection, ledPath, "xyz.openbmc_project.Led.Physical", "State",
1380 std::move(respHandler));
1381 }
1382
1383 BMCWEB_LOG_DEBUG("getInventoryLedData exit");
1384 }
1385
1386 /**
1387 * @brief Gets LED data for LEDs associated with given inventory items.
1388 *
1389 * Gets the D-Bus connections (services) that provide LED data for the LEDs
1390 * associated with the specified inventory items. Then gets the LED data from
1391 * each connection and stores it in the inventory item.
1392 *
1393 * This data is later used to provide sensor property values in the JSON
1394 * response.
1395 *
1396 * Finds the LED data asynchronously. Invokes callback when information has
1397 * been obtained.
1398 *
1399 * The callback must have the following signature:
1400 * @code
1401 * callback()
1402 * @endcode
1403 *
1404 * @param sensorsAsyncResp Pointer to object holding response data.
1405 * @param inventoryItems D-Bus inventory items associated with sensors.
1406 * @param callback Callback to invoke when inventory items have been obtained.
1407 */
1408 template <typename Callback>
getInventoryLeds(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::vector<InventoryItem>> & inventoryItems,Callback && callback)1409 void getInventoryLeds(
1410 const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1411 const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
1412 Callback&& callback)
1413 {
1414 BMCWEB_LOG_DEBUG("getInventoryLeds enter");
1415
1416 const std::string path = "/xyz/openbmc_project";
1417 constexpr std::array<std::string_view, 1> interfaces = {
1418 "xyz.openbmc_project.Led.Physical"};
1419
1420 // Make call to ObjectMapper to find all inventory items
1421 dbus::utility::getSubTree(
1422 path, 0, interfaces,
1423 [callback = std::forward<Callback>(callback), sensorsAsyncResp,
1424 inventoryItems](
1425 const boost::system::error_code& ec,
1426 const dbus::utility::MapperGetSubTreeResponse& subtree) mutable {
1427 // Response handler for parsing output from GetSubTree
1428 BMCWEB_LOG_DEBUG("getInventoryLeds respHandler enter");
1429 if (ec)
1430 {
1431 messages::internalError(sensorsAsyncResp->asyncResp->res);
1432 BMCWEB_LOG_ERROR("getInventoryLeds respHandler DBus error {}",
1433 ec);
1434 return;
1435 }
1436
1437 // Build map of LED object paths to connections
1438 std::shared_ptr<std::map<std::string, std::string>> ledConnections =
1439 std::make_shared<std::map<std::string, std::string>>();
1440
1441 // Loop through objects from GetSubTree
1442 for (const std::pair<std::string,
1443 std::vector<std::pair<
1444 std::string, std::vector<std::string>>>>&
1445 object : subtree)
1446 {
1447 // Check if object path is LED for one of the specified
1448 // inventory items
1449 const std::string& ledPath = object.first;
1450 if (findInventoryItemForLed(*inventoryItems, ledPath) !=
1451 nullptr)
1452 {
1453 // Add mapping from ledPath to connection
1454 const std::string& connection =
1455 object.second.begin()->first;
1456 (*ledConnections)[ledPath] = connection;
1457 BMCWEB_LOG_DEBUG("Added mapping {} -> {}", ledPath,
1458 connection);
1459 }
1460 }
1461
1462 getInventoryLedData(sensorsAsyncResp, inventoryItems,
1463 ledConnections, std::move(callback));
1464 BMCWEB_LOG_DEBUG("getInventoryLeds respHandler exit");
1465 });
1466 BMCWEB_LOG_DEBUG("getInventoryLeds exit");
1467 }
1468
1469 /**
1470 * @brief Gets D-Bus data for Power Supply Attributes such as EfficiencyPercent
1471 *
1472 * Uses the specified connections (services) (currently assumes just one) to
1473 * obtain D-Bus data for Power Supply Attributes. Stores the resulting data in
1474 * the inventoryItems vector. Only stores data in Power Supply inventoryItems.
1475 *
1476 * This data is later used to provide sensor property values in the JSON
1477 * response.
1478 *
1479 * Finds the Power Supply Attributes data asynchronously. Invokes callback
1480 * when data has been obtained.
1481 *
1482 * The callback must have the following signature:
1483 * @code
1484 * callback(std::shared_ptr<std::vector<InventoryItem>> inventoryItems)
1485 * @endcode
1486 *
1487 * @param sensorsAsyncResp Pointer to object holding response data.
1488 * @param inventoryItems D-Bus inventory items associated with sensors.
1489 * @param psAttributesConnections Connections that provide data for the Power
1490 * Supply Attributes
1491 * @param callback Callback to invoke when data has been obtained.
1492 */
1493 template <typename Callback>
getPowerSupplyAttributesData(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::vector<InventoryItem>> & inventoryItems,const std::map<std::string,std::string> & psAttributesConnections,Callback && callback)1494 void getPowerSupplyAttributesData(
1495 const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1496 const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
1497 const std::map<std::string, std::string>& psAttributesConnections,
1498 Callback&& callback)
1499 {
1500 BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData enter");
1501
1502 if (psAttributesConnections.empty())
1503 {
1504 BMCWEB_LOG_DEBUG("Can't find PowerSupplyAttributes, no connections!");
1505 callback(inventoryItems);
1506 return;
1507 }
1508
1509 // Assuming just one connection (service) for now
1510 auto it = psAttributesConnections.begin();
1511
1512 const std::string& psAttributesPath = (*it).first;
1513 const std::string& psAttributesConnection = (*it).second;
1514
1515 // Response handler for Get DeratingFactor property
1516 auto respHandler = [sensorsAsyncResp, inventoryItems,
1517 callback = std::forward<Callback>(callback)](
1518 const boost::system::error_code& ec,
1519 uint32_t value) mutable {
1520 BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData respHandler enter");
1521 if (ec)
1522 {
1523 BMCWEB_LOG_ERROR(
1524 "getPowerSupplyAttributesData respHandler DBus error {}", ec);
1525 messages::internalError(sensorsAsyncResp->asyncResp->res);
1526 return;
1527 }
1528
1529 BMCWEB_LOG_DEBUG("PS EfficiencyPercent value: {}", value);
1530 // Store value in Power Supply Inventory Items
1531 for (InventoryItem& inventoryItem : *inventoryItems)
1532 {
1533 if (inventoryItem.isPowerSupply)
1534 {
1535 inventoryItem.powerSupplyEfficiencyPercent =
1536 static_cast<int>(value);
1537 }
1538 }
1539
1540 BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData respHandler exit");
1541 callback(inventoryItems);
1542 };
1543
1544 // Get the DeratingFactor property for the PowerSupplyAttributes
1545 // Currently only property on the interface/only one we care about
1546 dbus::utility::getProperty<uint32_t>(
1547 psAttributesConnection, psAttributesPath,
1548 "xyz.openbmc_project.Control.PowerSupplyAttributes", "DeratingFactor",
1549 std::move(respHandler));
1550
1551 BMCWEB_LOG_DEBUG("getPowerSupplyAttributesData exit");
1552 }
1553
1554 /**
1555 * @brief Gets the Power Supply Attributes such as EfficiencyPercent
1556 *
1557 * Gets the D-Bus connection (service) that provides Power Supply Attributes
1558 * data. Then gets the Power Supply Attributes data from the connection
1559 * (currently just assumes 1 connection) and stores the data in the inventory
1560 * item.
1561 *
1562 * This data is later used to provide sensor property values in the JSON
1563 * response. DeratingFactor on D-Bus is mapped to EfficiencyPercent on Redfish.
1564 *
1565 * Finds the Power Supply Attributes data asynchronously. Invokes callback
1566 * when information has been obtained.
1567 *
1568 * The callback must have the following signature:
1569 * @code
1570 * callback(std::shared_ptr<std::vector<InventoryItem>> inventoryItems)
1571 * @endcode
1572 *
1573 * @param sensorsAsyncResp Pointer to object holding response data.
1574 * @param inventoryItems D-Bus inventory items associated with sensors.
1575 * @param callback Callback to invoke when data has been obtained.
1576 */
1577 template <typename Callback>
getPowerSupplyAttributes(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::vector<InventoryItem>> & inventoryItems,Callback && callback)1578 void getPowerSupplyAttributes(
1579 const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1580 const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems,
1581 Callback&& callback)
1582 {
1583 BMCWEB_LOG_DEBUG("getPowerSupplyAttributes enter");
1584
1585 // Only need the power supply attributes when the Power Schema
1586 if (sensorsAsyncResp->chassisSubNode != sensors::powerNodeStr)
1587 {
1588 BMCWEB_LOG_DEBUG("getPowerSupplyAttributes exit since not Power");
1589 callback(inventoryItems);
1590 return;
1591 }
1592
1593 constexpr std::array<std::string_view, 1> interfaces = {
1594 "xyz.openbmc_project.Control.PowerSupplyAttributes"};
1595
1596 // Make call to ObjectMapper to find the PowerSupplyAttributes service
1597 dbus::utility::getSubTree(
1598 "/xyz/openbmc_project", 0, interfaces,
1599 [callback = std::forward<Callback>(callback), sensorsAsyncResp,
1600 inventoryItems](
1601 const boost::system::error_code& ec,
1602 const dbus::utility::MapperGetSubTreeResponse& subtree) mutable {
1603 // Response handler for parsing output from GetSubTree
1604 BMCWEB_LOG_DEBUG("getPowerSupplyAttributes respHandler enter");
1605 if (ec)
1606 {
1607 messages::internalError(sensorsAsyncResp->asyncResp->res);
1608 BMCWEB_LOG_ERROR(
1609 "getPowerSupplyAttributes respHandler DBus error {}", ec);
1610 return;
1611 }
1612 if (subtree.empty())
1613 {
1614 BMCWEB_LOG_DEBUG("Can't find Power Supply Attributes!");
1615 callback(inventoryItems);
1616 return;
1617 }
1618
1619 // Currently we only support 1 power supply attribute, use this for
1620 // all the power supplies. Build map of object path to connection.
1621 // Assume just 1 connection and 1 path for now.
1622 std::map<std::string, std::string> psAttributesConnections;
1623
1624 if (subtree[0].first.empty() || subtree[0].second.empty())
1625 {
1626 BMCWEB_LOG_DEBUG("Power Supply Attributes mapper error!");
1627 callback(inventoryItems);
1628 return;
1629 }
1630
1631 const std::string& psAttributesPath = subtree[0].first;
1632 const std::string& connection = subtree[0].second.begin()->first;
1633
1634 if (connection.empty())
1635 {
1636 BMCWEB_LOG_DEBUG("Power Supply Attributes mapper error!");
1637 callback(inventoryItems);
1638 return;
1639 }
1640
1641 psAttributesConnections[psAttributesPath] = connection;
1642 BMCWEB_LOG_DEBUG("Added mapping {} -> {}", psAttributesPath,
1643 connection);
1644
1645 getPowerSupplyAttributesData(sensorsAsyncResp, inventoryItems,
1646 psAttributesConnections,
1647 std::move(callback));
1648 BMCWEB_LOG_DEBUG("getPowerSupplyAttributes respHandler exit");
1649 });
1650 BMCWEB_LOG_DEBUG("getPowerSupplyAttributes exit");
1651 }
1652
1653 /**
1654 * @brief Gets inventory items associated with sensors.
1655 *
1656 * Finds the inventory items that are associated with the specified sensors.
1657 * Then gets D-Bus data for the inventory items, such as presence and VPD.
1658 *
1659 * This data is later used to provide sensor property values in the JSON
1660 * response.
1661 *
1662 * Finds the inventory items asynchronously. Invokes callback when the
1663 * inventory items have been obtained.
1664 *
1665 * The callback must have the following signature:
1666 * @code
1667 * callback(std::shared_ptr<std::vector<InventoryItem>> inventoryItems)
1668 * @endcode
1669 *
1670 * @param sensorsAsyncResp Pointer to object holding response data.
1671 * @param sensorNames All sensors within the current chassis.
1672 * implements ObjectManager.
1673 * @param callback Callback to invoke when inventory items have been obtained.
1674 */
1675 template <typename Callback>
1676 inline void
getInventoryItems(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::set<std::string>> & sensorNames,Callback && callback)1677 getInventoryItems(const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1678 const std::shared_ptr<std::set<std::string>>& sensorNames,
1679 Callback&& callback)
1680 {
1681 BMCWEB_LOG_DEBUG("getInventoryItems enter");
1682 auto getInventoryItemAssociationsCb =
1683 [sensorsAsyncResp, callback = std::forward<Callback>(callback)](
1684 const std::shared_ptr<std::vector<InventoryItem>>&
1685 inventoryItems) mutable {
1686 BMCWEB_LOG_DEBUG("getInventoryItemAssociationsCb enter");
1687 auto getInventoryItemsConnectionsCb =
1688 [sensorsAsyncResp, inventoryItems,
1689 callback = std::forward<Callback>(callback)](
1690 const std::shared_ptr<std::set<std::string>>&
1691 invConnections) mutable {
1692 BMCWEB_LOG_DEBUG("getInventoryItemsConnectionsCb enter");
1693 auto getInventoryItemsDataCb =
1694 [sensorsAsyncResp, inventoryItems,
1695 callback =
1696 std::forward<Callback>(callback)]() mutable {
1697 BMCWEB_LOG_DEBUG("getInventoryItemsDataCb enter");
1698
1699 auto getInventoryLedsCb =
1700 [sensorsAsyncResp, inventoryItems,
1701 callback = std::forward<Callback>(
1702 callback)]() mutable {
1703 BMCWEB_LOG_DEBUG(
1704 "getInventoryLedsCb enter");
1705 // Find Power Supply Attributes and get the
1706 // data
1707 getPowerSupplyAttributes(
1708 sensorsAsyncResp, inventoryItems,
1709 std::move(callback));
1710 BMCWEB_LOG_DEBUG("getInventoryLedsCb exit");
1711 };
1712
1713 // Find led connections and get the data
1714 getInventoryLeds(sensorsAsyncResp, inventoryItems,
1715 std::move(getInventoryLedsCb));
1716 BMCWEB_LOG_DEBUG("getInventoryItemsDataCb exit");
1717 };
1718
1719 // Get inventory item data from connections
1720 getInventoryItemsData(sensorsAsyncResp, inventoryItems,
1721 invConnections,
1722 std::move(getInventoryItemsDataCb));
1723 BMCWEB_LOG_DEBUG("getInventoryItemsConnectionsCb exit");
1724 };
1725
1726 // Get connections that provide inventory item data
1727 getInventoryItemsConnections(
1728 sensorsAsyncResp, inventoryItems,
1729 std::move(getInventoryItemsConnectionsCb));
1730 BMCWEB_LOG_DEBUG("getInventoryItemAssociationsCb exit");
1731 };
1732
1733 // Get associations from sensors to inventory items
1734 getInventoryItemAssociations(sensorsAsyncResp, sensorNames,
1735 std::move(getInventoryItemAssociationsCb));
1736 BMCWEB_LOG_DEBUG("getInventoryItems exit");
1737 }
1738
1739 /**
1740 * @brief Returns JSON PowerSupply object for the specified inventory item.
1741 *
1742 * Searches for a JSON PowerSupply object that matches the specified inventory
1743 * item. If one is not found, a new PowerSupply object is added to the JSON
1744 * array.
1745 *
1746 * Multiple sensors are often associated with one power supply inventory item.
1747 * As a result, multiple sensor values are stored in one JSON PowerSupply
1748 * object.
1749 *
1750 * @param powerSupplyArray JSON array containing Redfish PowerSupply objects.
1751 * @param inventoryItem Inventory item for the power supply.
1752 * @param chassisId Chassis that contains the power supply.
1753 * @return JSON PowerSupply object for the specified inventory item.
1754 */
getPowerSupply(nlohmann::json & powerSupplyArray,const InventoryItem & inventoryItem,const std::string & chassisId)1755 inline nlohmann::json& getPowerSupply(nlohmann::json& powerSupplyArray,
1756 const InventoryItem& inventoryItem,
1757 const std::string& chassisId)
1758 {
1759 std::string nameS;
1760 nameS.resize(inventoryItem.name.size());
1761 std::ranges::replace_copy(inventoryItem.name, nameS.begin(), '_', ' ');
1762 // Check if matching PowerSupply object already exists in JSON array
1763 for (nlohmann::json& powerSupply : powerSupplyArray)
1764 {
1765 nlohmann::json::iterator nameIt = powerSupply.find("Name");
1766 if (nameIt == powerSupply.end())
1767 {
1768 continue;
1769 }
1770 const std::string* name = nameIt->get_ptr<std::string*>();
1771 if (name == nullptr)
1772 {
1773 continue;
1774 }
1775 if (nameS == *name)
1776 {
1777 return powerSupply;
1778 }
1779 }
1780
1781 // Add new PowerSupply object to JSON array
1782 powerSupplyArray.push_back({});
1783 nlohmann::json& powerSupply = powerSupplyArray.back();
1784 boost::urls::url url =
1785 boost::urls::format("/redfish/v1/Chassis/{}/Power", chassisId);
1786 url.set_fragment(("/PowerSupplies"_json_pointer).to_string());
1787 powerSupply["@odata.id"] = std::move(url);
1788 std::string escaped;
1789 escaped.resize(inventoryItem.name.size());
1790 std::ranges::replace_copy(inventoryItem.name, escaped.begin(), '_', ' ');
1791 powerSupply["Name"] = std::move(escaped);
1792 powerSupply["Manufacturer"] = inventoryItem.manufacturer;
1793 powerSupply["Model"] = inventoryItem.model;
1794 powerSupply["PartNumber"] = inventoryItem.partNumber;
1795 powerSupply["SerialNumber"] = inventoryItem.serialNumber;
1796 sensor_utils::setLedState(powerSupply, &inventoryItem);
1797
1798 if (inventoryItem.powerSupplyEfficiencyPercent >= 0)
1799 {
1800 powerSupply["EfficiencyPercent"] =
1801 inventoryItem.powerSupplyEfficiencyPercent;
1802 }
1803
1804 powerSupply["Status"]["State"] =
1805 sensor_utils::getState(&inventoryItem, true);
1806 const char* health = inventoryItem.isFunctional ? "OK" : "Critical";
1807 powerSupply["Status"]["Health"] = health;
1808
1809 return powerSupply;
1810 }
1811
1812 /**
1813 * @brief Gets the values of the specified sensors.
1814 *
1815 * Stores the results as JSON in the SensorsAsyncResp.
1816 *
1817 * Gets the sensor values asynchronously. Stores the results later when the
1818 * information has been obtained.
1819 *
1820 * The sensorNames set contains all requested sensors for the current chassis.
1821 *
1822 * To minimize the number of DBus calls, the DBus method
1823 * org.freedesktop.DBus.ObjectManager.GetManagedObjects() is used to get the
1824 * values of all sensors provided by a connection (service).
1825 *
1826 * The connections set contains all the connections that provide sensor values.
1827 *
1828 * The InventoryItem vector contains D-Bus inventory items associated with the
1829 * sensors. Inventory item data is needed for some Redfish sensor properties.
1830 *
1831 * @param SensorsAsyncResp Pointer to object holding response data.
1832 * @param sensorNames All requested sensors within the current chassis.
1833 * @param connections Connections that provide sensor values.
1834 * implements ObjectManager.
1835 * @param inventoryItems Inventory items associated with the sensors.
1836 */
getSensorData(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::set<std::string>> & sensorNames,const std::set<std::string> & connections,const std::shared_ptr<std::vector<InventoryItem>> & inventoryItems)1837 inline void getSensorData(
1838 const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
1839 const std::shared_ptr<std::set<std::string>>& sensorNames,
1840 const std::set<std::string>& connections,
1841 const std::shared_ptr<std::vector<InventoryItem>>& inventoryItems)
1842 {
1843 BMCWEB_LOG_DEBUG("getSensorData enter");
1844 // Get managed objects from all services exposing sensors
1845 for (const std::string& connection : connections)
1846 {
1847 sdbusplus::message::object_path sensorPath(
1848 "/xyz/openbmc_project/sensors");
1849 dbus::utility::getManagedObjects(
1850 connection, sensorPath,
1851 [sensorsAsyncResp, sensorNames,
1852 inventoryItems](const boost::system::error_code& ec,
1853 const dbus::utility::ManagedObjectType& resp) {
1854 BMCWEB_LOG_DEBUG("getManagedObjectsCb enter");
1855 if (ec)
1856 {
1857 BMCWEB_LOG_ERROR("getManagedObjectsCb DBUS error: {}", ec);
1858 messages::internalError(sensorsAsyncResp->asyncResp->res);
1859 return;
1860 }
1861 auto chassisSubNode = sensor_utils::chassisSubNodeFromString(
1862 sensorsAsyncResp->chassisSubNode);
1863 // Go through all objects and update response with sensor data
1864 for (const auto& objDictEntry : resp)
1865 {
1866 const std::string& objPath =
1867 static_cast<const std::string&>(objDictEntry.first);
1868 BMCWEB_LOG_DEBUG("getManagedObjectsCb parsing object {}",
1869 objPath);
1870
1871 std::vector<std::string> split;
1872 // Reserve space for
1873 // /xyz/openbmc_project/sensors/<name>/<subname>
1874 split.reserve(6);
1875 // NOLINTNEXTLINE
1876 bmcweb::split(split, objPath, '/');
1877 if (split.size() < 6)
1878 {
1879 BMCWEB_LOG_ERROR("Got path that isn't long enough {}",
1880 objPath);
1881 continue;
1882 }
1883 // These indexes aren't intuitive, as split puts an empty
1884 // string at the beginning
1885 const std::string& sensorType = split[4];
1886 const std::string& sensorName = split[5];
1887 BMCWEB_LOG_DEBUG("sensorName {} sensorType {}", sensorName,
1888 sensorType);
1889 if (sensorNames->find(objPath) == sensorNames->end())
1890 {
1891 BMCWEB_LOG_DEBUG("{} not in sensor list ", sensorName);
1892 continue;
1893 }
1894
1895 // Find inventory item (if any) associated with sensor
1896 InventoryItem* inventoryItem =
1897 findInventoryItemForSensor(inventoryItems, objPath);
1898
1899 const std::string& sensorSchema =
1900 sensorsAsyncResp->chassisSubNode;
1901
1902 nlohmann::json* sensorJson = nullptr;
1903
1904 if (sensorSchema == sensors::sensorsNodeStr &&
1905 !sensorsAsyncResp->efficientExpand)
1906 {
1907 std::string sensorId =
1908 redfish::sensor_utils::getSensorId(sensorName,
1909 sensorType);
1910
1911 sensorsAsyncResp->asyncResp->res
1912 .jsonValue["@odata.id"] = boost::urls::format(
1913 "/redfish/v1/Chassis/{}/{}/{}",
1914 sensorsAsyncResp->chassisId,
1915 sensorsAsyncResp->chassisSubNode, sensorId);
1916 sensorJson =
1917 &(sensorsAsyncResp->asyncResp->res.jsonValue);
1918 }
1919 else
1920 {
1921 std::string fieldName;
1922 if (sensorsAsyncResp->efficientExpand)
1923 {
1924 fieldName = "Members";
1925 }
1926 else if (sensorType == "temperature")
1927 {
1928 fieldName = "Temperatures";
1929 }
1930 else if (sensorType == "fan" ||
1931 sensorType == "fan_tach" ||
1932 sensorType == "fan_pwm")
1933 {
1934 fieldName = "Fans";
1935 }
1936 else if (sensorType == "voltage")
1937 {
1938 fieldName = "Voltages";
1939 }
1940 else if (sensorType == "power")
1941 {
1942 if (sensorName == "total_power")
1943 {
1944 fieldName = "PowerControl";
1945 }
1946 else if ((inventoryItem != nullptr) &&
1947 (inventoryItem->isPowerSupply))
1948 {
1949 fieldName = "PowerSupplies";
1950 }
1951 else
1952 {
1953 // Other power sensors are in SensorCollection
1954 continue;
1955 }
1956 }
1957 else
1958 {
1959 BMCWEB_LOG_ERROR(
1960 "Unsure how to handle sensorType {}",
1961 sensorType);
1962 continue;
1963 }
1964
1965 nlohmann::json& tempArray =
1966 sensorsAsyncResp->asyncResp->res
1967 .jsonValue[fieldName];
1968 if (fieldName == "PowerControl")
1969 {
1970 if (tempArray.empty())
1971 {
1972 // Put multiple "sensors" into a single
1973 // PowerControl. Follows MemberId naming and
1974 // naming in power.hpp.
1975 nlohmann::json::object_t power;
1976 boost::urls::url url = boost::urls::format(
1977 "/redfish/v1/Chassis/{}/{}",
1978 sensorsAsyncResp->chassisId,
1979 sensorsAsyncResp->chassisSubNode);
1980 url.set_fragment(
1981 (""_json_pointer / fieldName / "0")
1982 .to_string());
1983 power["@odata.id"] = std::move(url);
1984 tempArray.emplace_back(std::move(power));
1985 }
1986 sensorJson = &(tempArray.back());
1987 }
1988 else if (fieldName == "PowerSupplies")
1989 {
1990 if (inventoryItem != nullptr)
1991 {
1992 sensorJson = &(getPowerSupply(
1993 tempArray, *inventoryItem,
1994 sensorsAsyncResp->chassisId));
1995 }
1996 }
1997 else if (fieldName == "Members")
1998 {
1999 std::string sensorId =
2000 redfish::sensor_utils::getSensorId(sensorName,
2001 sensorType);
2002
2003 nlohmann::json::object_t member;
2004 member["@odata.id"] = boost::urls::format(
2005 "/redfish/v1/Chassis/{}/{}/{}",
2006 sensorsAsyncResp->chassisId,
2007 sensorsAsyncResp->chassisSubNode, sensorId);
2008 tempArray.emplace_back(std::move(member));
2009 sensorJson = &(tempArray.back());
2010 }
2011 else
2012 {
2013 nlohmann::json::object_t member;
2014 boost::urls::url url = boost::urls::format(
2015 "/redfish/v1/Chassis/{}/{}",
2016 sensorsAsyncResp->chassisId,
2017 sensorsAsyncResp->chassisSubNode);
2018 url.set_fragment(
2019 (""_json_pointer / fieldName).to_string());
2020 member["@odata.id"] = std::move(url);
2021 tempArray.emplace_back(std::move(member));
2022 sensorJson = &(tempArray.back());
2023 }
2024 }
2025
2026 if (sensorJson != nullptr)
2027 {
2028 objectInterfacesToJson(
2029 sensorName, sensorType, chassisSubNode,
2030 objDictEntry.second, *sensorJson, inventoryItem);
2031
2032 std::string path = "/xyz/openbmc_project/sensors/";
2033 path += sensorType;
2034 path += "/";
2035 path += sensorName;
2036 sensorsAsyncResp->addMetadata(*sensorJson, path);
2037 }
2038 }
2039 if (sensorsAsyncResp.use_count() == 1)
2040 {
2041 sortJSONResponse(sensorsAsyncResp);
2042 if (chassisSubNode ==
2043 sensor_utils::ChassisSubNode::sensorsNode &&
2044 sensorsAsyncResp->efficientExpand)
2045 {
2046 sensorsAsyncResp->asyncResp->res
2047 .jsonValue["Members@odata.count"] =
2048 sensorsAsyncResp->asyncResp->res
2049 .jsonValue["Members"]
2050 .size();
2051 }
2052 else if (chassisSubNode ==
2053 sensor_utils::ChassisSubNode::thermalNode)
2054 {
2055 populateFanRedundancy(sensorsAsyncResp);
2056 }
2057 }
2058 BMCWEB_LOG_DEBUG("getManagedObjectsCb exit");
2059 });
2060 }
2061 BMCWEB_LOG_DEBUG("getSensorData exit");
2062 }
2063
2064 inline void
processSensorList(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp,const std::shared_ptr<std::set<std::string>> & sensorNames)2065 processSensorList(const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
2066 const std::shared_ptr<std::set<std::string>>& sensorNames)
2067 {
2068 auto getConnectionCb = [sensorsAsyncResp, sensorNames](
2069 const std::set<std::string>& connections) {
2070 BMCWEB_LOG_DEBUG("getConnectionCb enter");
2071 auto getInventoryItemsCb =
2072 [sensorsAsyncResp, sensorNames, connections](
2073 const std::shared_ptr<std::vector<InventoryItem>>&
2074 inventoryItems) mutable {
2075 BMCWEB_LOG_DEBUG("getInventoryItemsCb enter");
2076 // Get sensor data and store results in JSON
2077 getSensorData(sensorsAsyncResp, sensorNames, connections,
2078 inventoryItems);
2079 BMCWEB_LOG_DEBUG("getInventoryItemsCb exit");
2080 };
2081
2082 // Get inventory items associated with sensors
2083 getInventoryItems(sensorsAsyncResp, sensorNames,
2084 std::move(getInventoryItemsCb));
2085
2086 BMCWEB_LOG_DEBUG("getConnectionCb exit");
2087 };
2088
2089 // Get set of connections that provide sensor values
2090 getConnections(sensorsAsyncResp, sensorNames, std::move(getConnectionCb));
2091 }
2092
2093 /**
2094 * @brief Entry point for retrieving sensors data related to requested
2095 * chassis.
2096 * @param SensorsAsyncResp Pointer to object holding response data
2097 */
2098 inline void
getChassisData(const std::shared_ptr<SensorsAsyncResp> & sensorsAsyncResp)2099 getChassisData(const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp)
2100 {
2101 BMCWEB_LOG_DEBUG("getChassisData enter");
2102 auto getChassisCb =
2103 [sensorsAsyncResp](
2104 const std::shared_ptr<std::set<std::string>>& sensorNames) {
2105 BMCWEB_LOG_DEBUG("getChassisCb enter");
2106 processSensorList(sensorsAsyncResp, sensorNames);
2107 BMCWEB_LOG_DEBUG("getChassisCb exit");
2108 };
2109 // SensorCollection doesn't contain the Redundancy property
2110 if (sensorsAsyncResp->chassisSubNode != sensors::sensorsNodeStr)
2111 {
2112 sensorsAsyncResp->asyncResp->res.jsonValue["Redundancy"] =
2113 nlohmann::json::array();
2114 }
2115 // Get set of sensors in chassis
2116 getChassis(sensorsAsyncResp->asyncResp, sensorsAsyncResp->chassisId,
2117 sensorsAsyncResp->chassisSubNode, sensorsAsyncResp->types,
2118 std::move(getChassisCb));
2119 BMCWEB_LOG_DEBUG("getChassisData exit");
2120 }
2121
2122 /**
2123 * @brief Find the requested sensorName in the list of all sensors supplied by
2124 * the chassis node
2125 *
2126 * @param sensorName The sensor name supplied in the PATCH request
2127 * @param sensorsList The list of sensors managed by the chassis node
2128 * @param sensorsModified The list of sensors that were found as a result of
2129 * repeated calls to this function
2130 */
findSensorNameUsingSensorPath(std::string_view sensorName,const std::set<std::string> & sensorsList,std::set<std::string> & sensorsModified)2131 inline bool findSensorNameUsingSensorPath(
2132 std::string_view sensorName, const std::set<std::string>& sensorsList,
2133 std::set<std::string>& sensorsModified)
2134 {
2135 for (const auto& chassisSensor : sensorsList)
2136 {
2137 sdbusplus::message::object_path path(chassisSensor);
2138 std::string thisSensorName = path.filename();
2139 if (thisSensorName.empty())
2140 {
2141 continue;
2142 }
2143 if (thisSensorName == sensorName)
2144 {
2145 sensorsModified.emplace(chassisSensor);
2146 return true;
2147 }
2148 }
2149 return false;
2150 }
2151
2152 /**
2153 * @brief Entry point for overriding sensor values of given sensor
2154 *
2155 * @param sensorAsyncResp response object
2156 * @param allCollections Collections extract from sensors' request patch info
2157 * @param chassisSubNode Chassis Node for which the query has to happen
2158 */
setSensorsOverride(const std::shared_ptr<SensorsAsyncResp> & sensorAsyncResp,std::unordered_map<std::string,std::vector<nlohmann::json::object_t>> & allCollections)2159 inline void setSensorsOverride(
2160 const std::shared_ptr<SensorsAsyncResp>& sensorAsyncResp,
2161 std::unordered_map<std::string, std::vector<nlohmann::json::object_t>>&
2162 allCollections)
2163 {
2164 BMCWEB_LOG_INFO("setSensorsOverride for subNode{}",
2165 sensorAsyncResp->chassisSubNode);
2166
2167 std::string_view propertyValueName;
2168 std::unordered_map<std::string, std::pair<double, std::string>> overrideMap;
2169 std::string memberId;
2170 double value = 0.0;
2171 for (auto& collectionItems : allCollections)
2172 {
2173 if (collectionItems.first == "Temperatures")
2174 {
2175 propertyValueName = "ReadingCelsius";
2176 }
2177 else if (collectionItems.first == "Fans")
2178 {
2179 propertyValueName = "Reading";
2180 }
2181 else
2182 {
2183 propertyValueName = "ReadingVolts";
2184 }
2185 for (auto& item : collectionItems.second)
2186 {
2187 if (!json_util::readJsonObject( //
2188 item, sensorAsyncResp->asyncResp->res, //
2189 "MemberId", memberId, //
2190 propertyValueName, value //
2191 ))
2192 {
2193 return;
2194 }
2195 overrideMap.emplace(memberId,
2196 std::make_pair(value, collectionItems.first));
2197 }
2198 }
2199
2200 auto getChassisSensorListCb = [sensorAsyncResp, overrideMap,
2201 propertyValueNameStr =
2202 std::string(propertyValueName)](
2203 const std::shared_ptr<
2204 std::set<std::string>>& sensorsList) {
2205 // Match sensor names in the PATCH request to those managed by the
2206 // chassis node
2207 const std::shared_ptr<std::set<std::string>> sensorNames =
2208 std::make_shared<std::set<std::string>>();
2209 for (const auto& item : overrideMap)
2210 {
2211 const auto& sensor = item.first;
2212 std::pair<std::string, std::string> sensorNameType =
2213 redfish::sensor_utils::splitSensorNameAndType(sensor);
2214 if (!findSensorNameUsingSensorPath(sensorNameType.second,
2215 *sensorsList, *sensorNames))
2216 {
2217 BMCWEB_LOG_INFO("Unable to find memberId {}", item.first);
2218 messages::resourceNotFound(sensorAsyncResp->asyncResp->res,
2219 item.second.second, item.first);
2220 return;
2221 }
2222 }
2223 // Get the connection to which the memberId belongs
2224 auto getObjectsWithConnectionCb = [sensorAsyncResp, overrideMap,
2225 propertyValueNameStr](
2226 const std::set<
2227 std::string>& /*connections*/,
2228 const std::set<std::pair<
2229 std::string, std::string>>&
2230 objectsWithConnection) {
2231 if (objectsWithConnection.size() != overrideMap.size())
2232 {
2233 BMCWEB_LOG_INFO(
2234 "Unable to find all objects with proper connection {} requested {}",
2235 objectsWithConnection.size(), overrideMap.size());
2236 messages::resourceNotFound(
2237 sensorAsyncResp->asyncResp->res,
2238 sensorAsyncResp->chassisSubNode == sensors::thermalNodeStr
2239 ? "Temperatures"
2240 : "Voltages",
2241 "Count");
2242 return;
2243 }
2244 for (const auto& item : objectsWithConnection)
2245 {
2246 sdbusplus::message::object_path path(item.first);
2247 std::string sensorName = path.filename();
2248 if (sensorName.empty())
2249 {
2250 messages::internalError(sensorAsyncResp->asyncResp->res);
2251 return;
2252 }
2253 std::string id = redfish::sensor_utils::getSensorId(
2254 sensorName, path.parent_path().filename());
2255
2256 const auto& iterator = overrideMap.find(id);
2257 if (iterator == overrideMap.end())
2258 {
2259 BMCWEB_LOG_INFO("Unable to find sensor object{}",
2260 item.first);
2261 messages::internalError(sensorAsyncResp->asyncResp->res);
2262 return;
2263 }
2264 setDbusProperty(sensorAsyncResp->asyncResp,
2265 propertyValueNameStr, item.second, item.first,
2266 "xyz.openbmc_project.Sensor.Value", "Value",
2267 iterator->second.first);
2268 }
2269 };
2270 // Get object with connection for the given sensor name
2271 getObjectsWithConnection(sensorAsyncResp, sensorNames,
2272 std::move(getObjectsWithConnectionCb));
2273 };
2274 // get full sensor list for the given chassisId and cross verify the sensor.
2275 getChassis(sensorAsyncResp->asyncResp, sensorAsyncResp->chassisId,
2276 sensorAsyncResp->chassisSubNode, sensorAsyncResp->types,
2277 std::move(getChassisSensorListCb));
2278 }
2279
2280 /**
2281 * @brief Retrieves mapping of Redfish URIs to sensor value property to D-Bus
2282 * path of the sensor.
2283 *
2284 * Function builds valid Redfish response for sensor query of given chassis and
2285 * node. It then builds metadata about Redfish<->D-Bus correlations and provides
2286 * it to caller in a callback.
2287 *
2288 * @param chassis Chassis for which retrieval should be performed
2289 * @param node Node (group) of sensors. See sensor_utils::node for supported
2290 * values
2291 * @param mapComplete Callback to be called with retrieval result
2292 */
2293 template <typename Callback>
retrieveUriToDbusMap(const std::string & chassis,const std::string & node,Callback && mapComplete)2294 inline void retrieveUriToDbusMap(
2295 const std::string& chassis, const std::string& node, Callback&& mapComplete)
2296 {
2297 decltype(sensors::paths)::const_iterator pathIt =
2298 std::find_if(sensors::paths.cbegin(), sensors::paths.cend(),
2299 [&node](auto&& val) { return val.first == node; });
2300 if (pathIt == sensors::paths.cend())
2301 {
2302 BMCWEB_LOG_ERROR("Wrong node provided : {}", node);
2303 std::map<std::string, std::string> noop;
2304 mapComplete(boost::beast::http::status::bad_request, noop);
2305 return;
2306 }
2307
2308 auto asyncResp = std::make_shared<bmcweb::AsyncResp>();
2309 auto callback =
2310 [asyncResp, mapCompleteCb = std::forward<Callback>(mapComplete)](
2311 const boost::beast::http::status status,
2312 const std::map<std::string, std::string>& uriToDbus) {
2313 mapCompleteCb(status, uriToDbus);
2314 };
2315
2316 auto resp = std::make_shared<SensorsAsyncResp>(
2317 asyncResp, chassis, pathIt->second, node, std::move(callback));
2318 getChassisData(resp);
2319 }
2320
2321 namespace sensors
2322 {
2323
getChassisCallback(const std::shared_ptr<bmcweb::AsyncResp> & asyncResp,std::string_view chassisId,std::string_view chassisSubNode,const std::shared_ptr<std::set<std::string>> & sensorNames)2324 inline void getChassisCallback(
2325 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2326 std::string_view chassisId, std::string_view chassisSubNode,
2327 const std::shared_ptr<std::set<std::string>>& sensorNames)
2328 {
2329 BMCWEB_LOG_DEBUG("getChassisCallback enter ");
2330
2331 nlohmann::json& entriesArray = asyncResp->res.jsonValue["Members"];
2332 for (const std::string& sensor : *sensorNames)
2333 {
2334 BMCWEB_LOG_DEBUG("Adding sensor: {}", sensor);
2335
2336 sdbusplus::message::object_path path(sensor);
2337 std::string sensorName = path.filename();
2338 if (sensorName.empty())
2339 {
2340 BMCWEB_LOG_ERROR("Invalid sensor path: {}", sensor);
2341 messages::internalError(asyncResp->res);
2342 return;
2343 }
2344 std::string type = path.parent_path().filename();
2345 std::string id = redfish::sensor_utils::getSensorId(sensorName, type);
2346
2347 nlohmann::json::object_t member;
2348 member["@odata.id"] = boost::urls::format(
2349 "/redfish/v1/Chassis/{}/{}/{}", chassisId, chassisSubNode, id);
2350
2351 entriesArray.emplace_back(std::move(member));
2352 }
2353
2354 asyncResp->res.jsonValue["Members@odata.count"] = entriesArray.size();
2355 BMCWEB_LOG_DEBUG("getChassisCallback exit");
2356 }
2357
handleSensorCollectionGet(App & app,const crow::Request & req,const std::shared_ptr<bmcweb::AsyncResp> & asyncResp,const std::string & chassisId)2358 inline void handleSensorCollectionGet(
2359 App& app, const crow::Request& req,
2360 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2361 const std::string& chassisId)
2362 {
2363 query_param::QueryCapabilities capabilities = {
2364 .canDelegateExpandLevel = 1,
2365 };
2366 query_param::Query delegatedQuery;
2367 if (!redfish::setUpRedfishRouteWithDelegation(app, req, asyncResp,
2368 delegatedQuery, capabilities))
2369 {
2370 return;
2371 }
2372
2373 if (delegatedQuery.expandType != query_param::ExpandType::None)
2374 {
2375 // we perform efficient expand.
2376 auto sensorsAsyncResp = std::make_shared<SensorsAsyncResp>(
2377 asyncResp, chassisId, sensors::dbus::sensorPaths,
2378 sensors::sensorsNodeStr,
2379 /*efficientExpand=*/true);
2380 getChassisData(sensorsAsyncResp);
2381
2382 BMCWEB_LOG_DEBUG(
2383 "SensorCollection doGet exit via efficient expand handler");
2384 return;
2385 }
2386
2387 // We get all sensors as hyperlinkes in the chassis (this
2388 // implies we reply on the default query parameters handler)
2389 getChassis(asyncResp, chassisId, sensors::sensorsNodeStr, dbus::sensorPaths,
2390 std::bind_front(sensors::getChassisCallback, asyncResp,
2391 chassisId, sensors::sensorsNodeStr));
2392 }
2393
2394 inline void
getSensorFromDbus(const std::shared_ptr<bmcweb::AsyncResp> & asyncResp,const std::string & sensorPath,const::dbus::utility::MapperGetObject & mapperResponse)2395 getSensorFromDbus(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2396 const std::string& sensorPath,
2397 const ::dbus::utility::MapperGetObject& mapperResponse)
2398 {
2399 if (mapperResponse.size() != 1)
2400 {
2401 messages::internalError(asyncResp->res);
2402 return;
2403 }
2404 const auto& valueIface = *mapperResponse.begin();
2405 const std::string& connectionName = valueIface.first;
2406 BMCWEB_LOG_DEBUG("Looking up {}", connectionName);
2407 BMCWEB_LOG_DEBUG("Path {}", sensorPath);
2408
2409 ::dbus::utility::getAllProperties(
2410 *crow::connections::systemBus, connectionName, sensorPath, "",
2411 [asyncResp,
2412 sensorPath](const boost::system::error_code& ec,
2413 const ::dbus::utility::DBusPropertiesMap& valuesDict) {
2414 if (ec)
2415 {
2416 messages::internalError(asyncResp->res);
2417 return;
2418 }
2419 sdbusplus::message::object_path path(sensorPath);
2420 std::string name = path.filename();
2421 path = path.parent_path();
2422 std::string type = path.filename();
2423 sensor_utils::objectPropertiesToJson(
2424 name, type, sensor_utils::ChassisSubNode::sensorsNode,
2425 valuesDict, asyncResp->res.jsonValue, nullptr);
2426 });
2427 }
2428
handleSensorGet(App & app,const crow::Request & req,const std::shared_ptr<bmcweb::AsyncResp> & asyncResp,const std::string & chassisId,const std::string & sensorId)2429 inline void handleSensorGet(App& app, const crow::Request& req,
2430 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
2431 const std::string& chassisId,
2432 const std::string& sensorId)
2433 {
2434 if (!redfish::setUpRedfishRoute(app, req, asyncResp))
2435 {
2436 return;
2437 }
2438 std::pair<std::string, std::string> nameType =
2439 redfish::sensor_utils::splitSensorNameAndType(sensorId);
2440 if (nameType.first.empty() || nameType.second.empty())
2441 {
2442 messages::resourceNotFound(asyncResp->res, sensorId, "Sensor");
2443 return;
2444 }
2445
2446 asyncResp->res.jsonValue["@odata.id"] = boost::urls::format(
2447 "/redfish/v1/Chassis/{}/Sensors/{}", chassisId, sensorId);
2448
2449 BMCWEB_LOG_DEBUG("Sensor doGet enter");
2450
2451 constexpr std::array<std::string_view, 1> interfaces = {
2452 "xyz.openbmc_project.Sensor.Value"};
2453 std::string sensorPath = "/xyz/openbmc_project/sensors/" + nameType.first +
2454 '/' + nameType.second;
2455 // Get a list of all of the sensors that implement Sensor.Value
2456 // and get the path and service name associated with the sensor
2457 ::dbus::utility::getDbusObject(
2458 sensorPath, interfaces,
2459 [asyncResp, sensorId,
2460 sensorPath](const boost::system::error_code& ec,
2461 const ::dbus::utility::MapperGetObject& subtree) {
2462 BMCWEB_LOG_DEBUG("respHandler1 enter");
2463 if (ec == boost::system::errc::io_error)
2464 {
2465 BMCWEB_LOG_WARNING("Sensor not found from getSensorPaths");
2466 messages::resourceNotFound(asyncResp->res, sensorId, "Sensor");
2467 return;
2468 }
2469 if (ec)
2470 {
2471 messages::internalError(asyncResp->res);
2472 BMCWEB_LOG_ERROR(
2473 "Sensor getSensorPaths resp_handler: Dbus error {}", ec);
2474 return;
2475 }
2476 getSensorFromDbus(asyncResp, sensorPath, subtree);
2477 BMCWEB_LOG_DEBUG("respHandler1 exit");
2478 });
2479 }
2480
2481 } // namespace sensors
2482
requestRoutesSensorCollection(App & app)2483 inline void requestRoutesSensorCollection(App& app)
2484 {
2485 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Sensors/")
2486 .privileges(redfish::privileges::getSensorCollection)
2487 .methods(boost::beast::http::verb::get)(
2488 std::bind_front(sensors::handleSensorCollectionGet, std::ref(app)));
2489 }
2490
requestRoutesSensor(App & app)2491 inline void requestRoutesSensor(App& app)
2492 {
2493 BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Sensors/<str>/")
2494 .privileges(redfish::privileges::getSensor)
2495 .methods(boost::beast::http::verb::get)(
2496 std::bind_front(sensors::handleSensorGet, std::ref(app)));
2497 }
2498
2499 } // namespace redfish
2500