xref: /openbmc/bmcweb/features/redfish/lib/sensors.hpp (revision e0d918bc397350aa21af3dab9faa6e21748f6373)
108777fb0SLewanczyk, Dawid /*
208777fb0SLewanczyk, Dawid // Copyright (c) 2018 Intel Corporation
308777fb0SLewanczyk, Dawid //
408777fb0SLewanczyk, Dawid // Licensed under the Apache License, Version 2.0 (the "License");
508777fb0SLewanczyk, Dawid // you may not use this file except in compliance with the License.
608777fb0SLewanczyk, Dawid // You may obtain a copy of the License at
708777fb0SLewanczyk, Dawid //
808777fb0SLewanczyk, Dawid //      http://www.apache.org/licenses/LICENSE-2.0
908777fb0SLewanczyk, Dawid //
1008777fb0SLewanczyk, Dawid // Unless required by applicable law or agreed to in writing, software
1108777fb0SLewanczyk, Dawid // distributed under the License is distributed on an "AS IS" BASIS,
1208777fb0SLewanczyk, Dawid // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1308777fb0SLewanczyk, Dawid // See the License for the specific language governing permissions and
1408777fb0SLewanczyk, Dawid // limitations under the License.
1508777fb0SLewanczyk, Dawid */
1608777fb0SLewanczyk, Dawid #pragma once
1708777fb0SLewanczyk, Dawid 
1808777fb0SLewanczyk, Dawid #include <math.h>
1908777fb0SLewanczyk, Dawid #include <dbus_singleton.hpp>
2008777fb0SLewanczyk, Dawid #include <boost/algorithm/string/predicate.hpp>
2108777fb0SLewanczyk, Dawid #include <boost/algorithm/string/split.hpp>
2208777fb0SLewanczyk, Dawid #include <boost/container/flat_map.hpp>
2308777fb0SLewanczyk, Dawid #include <boost/range/algorithm/replace_copy_if.hpp>
2408777fb0SLewanczyk, Dawid 
2508777fb0SLewanczyk, Dawid namespace redfish {
2608777fb0SLewanczyk, Dawid 
2708777fb0SLewanczyk, Dawid constexpr const char* DBUS_SENSOR_PREFIX = "/xyz/openbmc_project/Sensors/";
2808777fb0SLewanczyk, Dawid 
2908777fb0SLewanczyk, Dawid using GetSubTreeType = std::vector<
3008777fb0SLewanczyk, Dawid     std::pair<std::string,
3108777fb0SLewanczyk, Dawid               std::vector<std::pair<std::string, std::vector<std::string>>>>>;
3208777fb0SLewanczyk, Dawid 
33aa2e59c1SEd Tanous using SensorVariant = sdbusplus::message::variant<int64_t, double>;
34aa2e59c1SEd Tanous 
3508777fb0SLewanczyk, Dawid using ManagedObjectsVectorType = std::vector<std::pair<
36aa2e59c1SEd Tanous     sdbusplus::message::object_path,
3708777fb0SLewanczyk, Dawid     boost::container::flat_map<
38aa2e59c1SEd Tanous         std::string, boost::container::flat_map<std::string, SensorVariant>>>>;
3908777fb0SLewanczyk, Dawid 
4008777fb0SLewanczyk, Dawid /**
4108777fb0SLewanczyk, Dawid  * AsyncResp
4208777fb0SLewanczyk, Dawid  * Gathers data needed for response processing after async calls are done
4308777fb0SLewanczyk, Dawid  */
4408777fb0SLewanczyk, Dawid class AsyncResp {
4508777fb0SLewanczyk, Dawid  public:
4608777fb0SLewanczyk, Dawid   AsyncResp(crow::response& response, const std::string& chassisId,
4708777fb0SLewanczyk, Dawid             const std::initializer_list<const char*> types)
4808777fb0SLewanczyk, Dawid       : chassisId(chassisId), res(response), types(types) {
4908777fb0SLewanczyk, Dawid     res.json_value["@odata.id"] =
5008777fb0SLewanczyk, Dawid         "/redfish/v1/Chassis/" + chassisId + "/Thermal";
5108777fb0SLewanczyk, Dawid   }
5208777fb0SLewanczyk, Dawid 
5308777fb0SLewanczyk, Dawid   ~AsyncResp() {
54*e0d918bcSEd Tanous     if (res.result() == boost::beast::http::status::internal_server_error) {
5508777fb0SLewanczyk, Dawid       // Reset the json object to clear out any data that made it in before the
5608777fb0SLewanczyk, Dawid       // error happened
5708777fb0SLewanczyk, Dawid       // todo(ed) handle error condition with proper code
5808777fb0SLewanczyk, Dawid       res.json_value = nlohmann::json::object();
5908777fb0SLewanczyk, Dawid     }
6008777fb0SLewanczyk, Dawid     res.end();
6108777fb0SLewanczyk, Dawid   }
6208777fb0SLewanczyk, Dawid   void setErrorStatus() {
63*e0d918bcSEd Tanous     res.result(boost::beast::http::status::internal_server_error);
6408777fb0SLewanczyk, Dawid   }
6508777fb0SLewanczyk, Dawid 
6608777fb0SLewanczyk, Dawid   std::string chassisId{};
6708777fb0SLewanczyk, Dawid   crow::response& res;
6808777fb0SLewanczyk, Dawid   const std::vector<const char*> types;
6908777fb0SLewanczyk, Dawid };
7008777fb0SLewanczyk, Dawid 
7108777fb0SLewanczyk, Dawid /**
7208777fb0SLewanczyk, Dawid  * @brief Creates connections necessary for chassis sensors
7308777fb0SLewanczyk, Dawid  * @param asyncResp Pointer to object holding response data
7408777fb0SLewanczyk, Dawid  * @param sensorNames Sensors retrieved from chassis
7508777fb0SLewanczyk, Dawid  * @param callback Callback for processing gathered connections
7608777fb0SLewanczyk, Dawid  */
7708777fb0SLewanczyk, Dawid template <typename Callback>
78*e0d918bcSEd Tanous void getConnections(std::shared_ptr<AsyncResp> asyncResp,
7908777fb0SLewanczyk, Dawid                     const boost::container::flat_set<std::string>& sensorNames,
8008777fb0SLewanczyk, Dawid                     Callback&& callback) {
8108777fb0SLewanczyk, Dawid   CROW_LOG_DEBUG << "getConnections";
8208777fb0SLewanczyk, Dawid   const std::string path = "/xyz/openbmc_project/Sensors";
8308777fb0SLewanczyk, Dawid   const std::array<std::string, 1> interfaces = {
8408777fb0SLewanczyk, Dawid       "xyz.openbmc_project.Sensor.Value"};
8508777fb0SLewanczyk, Dawid 
8608777fb0SLewanczyk, Dawid   // Response handler for parsing objects subtree
8708777fb0SLewanczyk, Dawid   auto resp_handler = [ callback{std::move(callback)}, asyncResp, sensorNames ](
8808777fb0SLewanczyk, Dawid       const boost::system::error_code ec, const GetSubTreeType& subtree) {
89*e0d918bcSEd Tanous     if (ec) {
9008777fb0SLewanczyk, Dawid       asyncResp->setErrorStatus();
91daf36e2eSEd Tanous       CROW_LOG_ERROR << "resp_handler: Dbus error " << ec;
9208777fb0SLewanczyk, Dawid       return;
9308777fb0SLewanczyk, Dawid     }
9408777fb0SLewanczyk, Dawid 
9508777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "Found " << subtree.size() << " subtrees";
9608777fb0SLewanczyk, Dawid 
9708777fb0SLewanczyk, Dawid     // Make unique list of connections only for requested sensor types and
9808777fb0SLewanczyk, Dawid     // found in the chassis
9908777fb0SLewanczyk, Dawid     boost::container::flat_set<std::string> connections;
10008777fb0SLewanczyk, Dawid     // Intrinsic to avoid malloc.  Most systems will have < 8 sensor producers
10108777fb0SLewanczyk, Dawid     connections.reserve(8);
10208777fb0SLewanczyk, Dawid 
10308777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "sensorNames list cout: " << sensorNames.size();
10408777fb0SLewanczyk, Dawid     for (const std::string& tsensor : sensorNames) {
10508777fb0SLewanczyk, Dawid       CROW_LOG_DEBUG << "Sensor to find: " << tsensor;
10608777fb0SLewanczyk, Dawid     }
10708777fb0SLewanczyk, Dawid 
10808777fb0SLewanczyk, Dawid     for (const std::pair<
10908777fb0SLewanczyk, Dawid              std::string,
11008777fb0SLewanczyk, Dawid              std::vector<std::pair<std::string, std::vector<std::string>>>>&
11108777fb0SLewanczyk, Dawid              object : subtree) {
11208777fb0SLewanczyk, Dawid       for (const char* type : asyncResp->types) {
11308777fb0SLewanczyk, Dawid         if (boost::starts_with(object.first, type)) {
11408777fb0SLewanczyk, Dawid           auto lastPos = object.first.rfind('/');
11508777fb0SLewanczyk, Dawid           if (lastPos != std::string::npos) {
11608777fb0SLewanczyk, Dawid             std::string sensorName = object.first.substr(lastPos + 1);
11708777fb0SLewanczyk, Dawid 
11808777fb0SLewanczyk, Dawid             if (sensorNames.find(sensorName) != sensorNames.end()) {
11908777fb0SLewanczyk, Dawid               // For each connection name
12008777fb0SLewanczyk, Dawid               for (const std::pair<std::string, std::vector<std::string>>&
12108777fb0SLewanczyk, Dawid                        objData : object.second) {
12208777fb0SLewanczyk, Dawid                 connections.insert(objData.first);
12308777fb0SLewanczyk, Dawid               }
12408777fb0SLewanczyk, Dawid             }
12508777fb0SLewanczyk, Dawid           }
12608777fb0SLewanczyk, Dawid           break;
12708777fb0SLewanczyk, Dawid         }
12808777fb0SLewanczyk, Dawid       }
12908777fb0SLewanczyk, Dawid     }
13008777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "Found " << connections.size() << " connections";
13108777fb0SLewanczyk, Dawid     callback(std::move(connections));
13208777fb0SLewanczyk, Dawid   };
13308777fb0SLewanczyk, Dawid 
13408777fb0SLewanczyk, Dawid   // Make call to ObjectMapper to find all sensors objects
135aa2e59c1SEd Tanous   crow::connections::system_bus->async_method_call(
136daf36e2eSEd Tanous       std::move(resp_handler), "xyz.openbmc_project.ObjectMapper",
137aa2e59c1SEd Tanous       "/xyz/openbmc_project/object_mapper", "xyz.openbmc_project.ObjectMapper",
138aa2e59c1SEd Tanous       "GetSubTree", path, 2, interfaces);
13908777fb0SLewanczyk, Dawid }
14008777fb0SLewanczyk, Dawid 
14108777fb0SLewanczyk, Dawid /**
14208777fb0SLewanczyk, Dawid  * @brief Retrieves requested chassis sensors and redundancy data from DBus .
14308777fb0SLewanczyk, Dawid  * @param asyncResp   Pointer to object holding response data
14408777fb0SLewanczyk, Dawid  * @param callback  Callback for next step in gathered sensor processing
14508777fb0SLewanczyk, Dawid  */
14608777fb0SLewanczyk, Dawid template <typename Callback>
147*e0d918bcSEd Tanous void getChassis(std::shared_ptr<AsyncResp> asyncResp, Callback&& callback) {
14808777fb0SLewanczyk, Dawid   CROW_LOG_DEBUG << "getChassis Done";
14908777fb0SLewanczyk, Dawid 
15008777fb0SLewanczyk, Dawid   // Process response from EntityManager and extract chassis data
15108777fb0SLewanczyk, Dawid   auto resp_handler = [ callback{std::move(callback)}, asyncResp ](
15208777fb0SLewanczyk, Dawid       const boost::system::error_code ec, ManagedObjectsVectorType& resp) {
15308777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "getChassis resp_handler called back Done";
15408777fb0SLewanczyk, Dawid     if (ec) {
15508777fb0SLewanczyk, Dawid       CROW_LOG_ERROR << "getChassis resp_handler got error " << ec;
15608777fb0SLewanczyk, Dawid       asyncResp->setErrorStatus();
15708777fb0SLewanczyk, Dawid       return;
15808777fb0SLewanczyk, Dawid     }
15908777fb0SLewanczyk, Dawid     boost::container::flat_set<std::string> sensorNames;
16008777fb0SLewanczyk, Dawid 
161daf36e2eSEd Tanous     //   asyncResp->chassisId
162daf36e2eSEd Tanous     bool foundChassis = false;
163daf36e2eSEd Tanous     std::vector<std::string> split;
164daf36e2eSEd Tanous     // Reserve space for
165daf36e2eSEd Tanous     // /xyz/openbmc_project/inventory/<name>/<subname> + 3 subnames
166daf36e2eSEd Tanous     split.reserve(8);
167daf36e2eSEd Tanous 
168daf36e2eSEd Tanous     for (const auto& objDictEntry : resp) {
169daf36e2eSEd Tanous       const std::string& objectPath =
170daf36e2eSEd Tanous           static_cast<const std::string&>(objDictEntry.first);
171daf36e2eSEd Tanous       boost::algorithm::split(split, objectPath, boost::is_any_of("/"));
172daf36e2eSEd Tanous       if (split.size() < 2) {
173daf36e2eSEd Tanous         CROW_LOG_ERROR << "Got path that isn't long enough " << objectPath;
174daf36e2eSEd Tanous         split.clear();
175daf36e2eSEd Tanous         continue;
176daf36e2eSEd Tanous       }
177daf36e2eSEd Tanous       const std::string& sensorName = split.end()[-1];
178daf36e2eSEd Tanous       const std::string& chassisName = split.end()[-2];
179daf36e2eSEd Tanous 
180daf36e2eSEd Tanous       if (chassisName != asyncResp->chassisId) {
181daf36e2eSEd Tanous         split.clear();
182daf36e2eSEd Tanous         continue;
183daf36e2eSEd Tanous       }
184daf36e2eSEd Tanous       foundChassis = true;
18508777fb0SLewanczyk, Dawid       sensorNames.emplace(sensorName);
186daf36e2eSEd Tanous       split.clear();
18708777fb0SLewanczyk, Dawid     };
18808777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "Found " << sensorNames.size() << " Sensor names";
18908777fb0SLewanczyk, Dawid 
19008777fb0SLewanczyk, Dawid     if (!foundChassis) {
19108777fb0SLewanczyk, Dawid       CROW_LOG_INFO << "Unable to find chassis named " << asyncResp->chassisId;
192*e0d918bcSEd Tanous       asyncResp->res.result(boost::beast::http::status::not_found);
19308777fb0SLewanczyk, Dawid     } else {
19408777fb0SLewanczyk, Dawid       callback(sensorNames);
19508777fb0SLewanczyk, Dawid     }
19608777fb0SLewanczyk, Dawid   };
19708777fb0SLewanczyk, Dawid 
19808777fb0SLewanczyk, Dawid   // Make call to EntityManager to find all chassis objects
199aa2e59c1SEd Tanous   crow::connections::system_bus->async_method_call(
200aa2e59c1SEd Tanous       resp_handler, "xyz.openbmc_project.EntityManager",
201daf36e2eSEd Tanous       "/xyz/openbmc_project/inventory", "org.freedesktop.DBus.ObjectManager",
202daf36e2eSEd Tanous       "GetManagedObjects");
20308777fb0SLewanczyk, Dawid }
20408777fb0SLewanczyk, Dawid 
20508777fb0SLewanczyk, Dawid /**
20608777fb0SLewanczyk, Dawid  * @brief Builds a json sensor representation of a sensor.
20708777fb0SLewanczyk, Dawid  * @param sensorName  The name of the sensor to be built
208274fad5aSGunnar Mills  * @param sensorType  The type (temperature, fan_tach, etc) of the sensor to
20908777fb0SLewanczyk, Dawid  * build
21008777fb0SLewanczyk, Dawid  * @param interfacesDict  A dictionary of the interfaces and properties of said
21108777fb0SLewanczyk, Dawid  * interfaces to be built from
21208777fb0SLewanczyk, Dawid  * @param sensor_json  The json object to fill
21308777fb0SLewanczyk, Dawid  */
21408777fb0SLewanczyk, Dawid void objectInterfacesToJson(
21508777fb0SLewanczyk, Dawid     const std::string& sensorName, const std::string& sensorType,
21608777fb0SLewanczyk, Dawid     const boost::container::flat_map<
217aa2e59c1SEd Tanous         std::string, boost::container::flat_map<std::string, SensorVariant>>&
21808777fb0SLewanczyk, Dawid         interfacesDict,
21908777fb0SLewanczyk, Dawid     nlohmann::json& sensor_json) {
22008777fb0SLewanczyk, Dawid   // We need a value interface before we can do anything with it
22108777fb0SLewanczyk, Dawid   auto value_it = interfacesDict.find("xyz.openbmc_project.Sensor.Value");
22208777fb0SLewanczyk, Dawid   if (value_it == interfacesDict.end()) {
22308777fb0SLewanczyk, Dawid     CROW_LOG_ERROR << "Sensor doesn't have a value interface";
22408777fb0SLewanczyk, Dawid     return;
22508777fb0SLewanczyk, Dawid   }
22608777fb0SLewanczyk, Dawid 
22708777fb0SLewanczyk, Dawid   // Assume values exist as is (10^0 == 1) if no scale exists
22808777fb0SLewanczyk, Dawid   int64_t scaleMultiplier = 0;
22908777fb0SLewanczyk, Dawid 
23008777fb0SLewanczyk, Dawid   auto scale_it = value_it->second.find("Scale");
23108777fb0SLewanczyk, Dawid   // If a scale exists, pull value as int64, and use the scaling.
23208777fb0SLewanczyk, Dawid   if (scale_it != value_it->second.end()) {
233aa2e59c1SEd Tanous     const int64_t* int64Value =
234aa2e59c1SEd Tanous         mapbox::get_ptr<const int64_t>(scale_it->second);
23508777fb0SLewanczyk, Dawid     if (int64Value != nullptr) {
23608777fb0SLewanczyk, Dawid       scaleMultiplier = *int64Value;
23708777fb0SLewanczyk, Dawid     }
23808777fb0SLewanczyk, Dawid   }
23908777fb0SLewanczyk, Dawid 
24008777fb0SLewanczyk, Dawid   sensor_json["MemberId"] = sensorName;
24108777fb0SLewanczyk, Dawid   sensor_json["Name"] = sensorName;
24208777fb0SLewanczyk, Dawid   sensor_json["Status"]["State"] = "Enabled";
24308777fb0SLewanczyk, Dawid   sensor_json["Status"]["Health"] = "OK";
24408777fb0SLewanczyk, Dawid 
24508777fb0SLewanczyk, Dawid   // Parameter to set to override the type we get from dbus, and force it to
24608777fb0SLewanczyk, Dawid   // int, regardless of what is available.  This is used for schemas like fan,
24708777fb0SLewanczyk, Dawid   // that require integers, not floats.
24808777fb0SLewanczyk, Dawid   bool forceToInt = false;
24908777fb0SLewanczyk, Dawid 
25008777fb0SLewanczyk, Dawid   const char* unit = "Reading";
25108777fb0SLewanczyk, Dawid   if (sensorType == "temperature") {
25208777fb0SLewanczyk, Dawid     unit = "ReadingCelsius";
25308777fb0SLewanczyk, Dawid     // TODO(ed) Documentation says that path should be type fan_tach,
25408777fb0SLewanczyk, Dawid     // implementation seems to implement fan
255daf36e2eSEd Tanous   } else if (sensorType == "fan" || sensorType == "fan_type") {
25608777fb0SLewanczyk, Dawid     unit = "Reading";
25708777fb0SLewanczyk, Dawid     sensor_json["ReadingUnits"] = "RPM";
25808777fb0SLewanczyk, Dawid     forceToInt = true;
25908777fb0SLewanczyk, Dawid   } else if (sensorType == "voltage") {
26008777fb0SLewanczyk, Dawid     unit = "ReadingVolts";
26108777fb0SLewanczyk, Dawid   } else {
26208777fb0SLewanczyk, Dawid     CROW_LOG_ERROR << "Redfish cannot map object type for " << sensorName;
26308777fb0SLewanczyk, Dawid     return;
26408777fb0SLewanczyk, Dawid   }
26508777fb0SLewanczyk, Dawid   // Map of dbus interface name, dbus property name and redfish property_name
26608777fb0SLewanczyk, Dawid   std::vector<std::tuple<const char*, const char*, const char*>> properties;
26708777fb0SLewanczyk, Dawid   properties.reserve(7);
26808777fb0SLewanczyk, Dawid 
26908777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Value", "Value", unit);
27008777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Warning",
27108777fb0SLewanczyk, Dawid                           "WarningHigh", "UpperThresholdNonCritical");
27208777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Warning",
27308777fb0SLewanczyk, Dawid                           "WarningLow", "LowerThresholdNonCritical");
27408777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Critical",
27508777fb0SLewanczyk, Dawid                           "CriticalHigh", "UpperThresholdCritical");
27608777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Critical",
27708777fb0SLewanczyk, Dawid                           "CriticalLow", "LowerThresholdCritical");
27808777fb0SLewanczyk, Dawid 
27908777fb0SLewanczyk, Dawid   if (sensorType == "temperature") {
28008777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MinValue",
28108777fb0SLewanczyk, Dawid                             "MinReadingRangeTemp");
28208777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MaxValue",
28308777fb0SLewanczyk, Dawid                             "MaxReadingRangeTemp");
28408777fb0SLewanczyk, Dawid   } else {
28508777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MinValue",
28608777fb0SLewanczyk, Dawid                             "MinReadingRange");
28708777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MaxValue",
28808777fb0SLewanczyk, Dawid                             "MaxReadingRange");
28908777fb0SLewanczyk, Dawid   }
29008777fb0SLewanczyk, Dawid 
29108777fb0SLewanczyk, Dawid   for (const std::tuple<const char*, const char*, const char*>& p :
29208777fb0SLewanczyk, Dawid        properties) {
29308777fb0SLewanczyk, Dawid     auto interfaceProperties = interfacesDict.find(std::get<0>(p));
29408777fb0SLewanczyk, Dawid     if (interfaceProperties != interfacesDict.end()) {
29508777fb0SLewanczyk, Dawid       auto value_it = interfaceProperties->second.find(std::get<1>(p));
29608777fb0SLewanczyk, Dawid       if (value_it != interfaceProperties->second.end()) {
297aa2e59c1SEd Tanous         const SensorVariant& valueVariant = value_it->second;
29808777fb0SLewanczyk, Dawid         nlohmann::json& value_it = sensor_json[std::get<2>(p)];
299aa2e59c1SEd Tanous 
30008777fb0SLewanczyk, Dawid         // Attempt to pull the int64 directly
301aa2e59c1SEd Tanous         const int64_t* int64Value =
302aa2e59c1SEd Tanous             mapbox::get_ptr<const int64_t>(valueVariant);
30308777fb0SLewanczyk, Dawid 
30408777fb0SLewanczyk, Dawid         if (int64Value != nullptr) {
30508777fb0SLewanczyk, Dawid           if (forceToInt || scaleMultiplier >= 0) {
30608777fb0SLewanczyk, Dawid             value_it = *int64Value * std::pow(10, scaleMultiplier);
30708777fb0SLewanczyk, Dawid           } else {
30808777fb0SLewanczyk, Dawid             value_it = *int64Value *
30908777fb0SLewanczyk, Dawid                        std::pow(10, static_cast<double>(scaleMultiplier));
31008777fb0SLewanczyk, Dawid           }
31108777fb0SLewanczyk, Dawid         }
31208777fb0SLewanczyk, Dawid         // Attempt to pull the float directly
313aa2e59c1SEd Tanous         const double* doubleValue = mapbox::get_ptr<const double>(valueVariant);
31408777fb0SLewanczyk, Dawid 
31508777fb0SLewanczyk, Dawid         if (doubleValue != nullptr) {
31608777fb0SLewanczyk, Dawid           if (!forceToInt) {
31708777fb0SLewanczyk, Dawid             value_it = *doubleValue *
31808777fb0SLewanczyk, Dawid                        std::pow(10, static_cast<double>(scaleMultiplier));
31908777fb0SLewanczyk, Dawid           } else {
32008777fb0SLewanczyk, Dawid             value_it = static_cast<int64_t>(*doubleValue *
32108777fb0SLewanczyk, Dawid                                             std::pow(10, scaleMultiplier));
32208777fb0SLewanczyk, Dawid           }
32308777fb0SLewanczyk, Dawid         }
32408777fb0SLewanczyk, Dawid       }
32508777fb0SLewanczyk, Dawid     }
32608777fb0SLewanczyk, Dawid   }
327*e0d918bcSEd Tanous   CROW_LOG_DEBUG << "Added sensor " << sensorName;
32808777fb0SLewanczyk, Dawid }
32908777fb0SLewanczyk, Dawid 
33008777fb0SLewanczyk, Dawid /**
33108777fb0SLewanczyk, Dawid  * @brief Entry point for retrieving sensors data related to requested
33208777fb0SLewanczyk, Dawid  *        chassis.
33308777fb0SLewanczyk, Dawid  * @param asyncResp   Pointer to object holding response data
33408777fb0SLewanczyk, Dawid  */
335*e0d918bcSEd Tanous void getChassisData(std::shared_ptr<AsyncResp> asyncResp) {
33608777fb0SLewanczyk, Dawid   CROW_LOG_DEBUG << "getChassisData";
33708777fb0SLewanczyk, Dawid   auto getChassisCb = [&, asyncResp](boost::container::flat_set<std::string>&
33808777fb0SLewanczyk, Dawid                                          sensorNames) {
33908777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "getChassisCb Done";
340*e0d918bcSEd Tanous     auto getConnectionCb = [&, asyncResp, sensorNames](
341*e0d918bcSEd Tanous                                const boost::container::flat_set<std::string>&
342*e0d918bcSEd Tanous                                    connections) {
34308777fb0SLewanczyk, Dawid       CROW_LOG_DEBUG << "getConnectionCb Done";
34408777fb0SLewanczyk, Dawid       // Get managed objects from all services exposing sensors
34508777fb0SLewanczyk, Dawid       for (const std::string& connection : connections) {
34608777fb0SLewanczyk, Dawid         // Response handler to process managed objects
34708777fb0SLewanczyk, Dawid         auto getManagedObjectsCb = [&, asyncResp, sensorNames](
34808777fb0SLewanczyk, Dawid                                        const boost::system::error_code ec,
34908777fb0SLewanczyk, Dawid                                        ManagedObjectsVectorType& resp) {
35008777fb0SLewanczyk, Dawid           // Go through all objects and update response with
35108777fb0SLewanczyk, Dawid           // sensor data
35208777fb0SLewanczyk, Dawid           for (const auto& objDictEntry : resp) {
353aa2e59c1SEd Tanous             const std::string& objPath =
354daf36e2eSEd Tanous                 static_cast<const std::string&>(objDictEntry.first);
355*e0d918bcSEd Tanous             CROW_LOG_DEBUG << "getManagedObjectsCb parsing object " << objPath;
356*e0d918bcSEd Tanous 
35708777fb0SLewanczyk, Dawid             std::vector<std::string> split;
35808777fb0SLewanczyk, Dawid             // Reserve space for
35908777fb0SLewanczyk, Dawid             // /xyz/openbmc_project/Sensors/<name>/<subname>
36008777fb0SLewanczyk, Dawid             split.reserve(6);
36108777fb0SLewanczyk, Dawid             boost::algorithm::split(split, objPath, boost::is_any_of("/"));
36208777fb0SLewanczyk, Dawid             if (split.size() < 6) {
363*e0d918bcSEd Tanous               CROW_LOG_ERROR << "Got path that isn't long enough " << objPath;
36408777fb0SLewanczyk, Dawid               continue;
36508777fb0SLewanczyk, Dawid             }
36608777fb0SLewanczyk, Dawid             // These indexes aren't intuitive, as boost::split puts an empty
367*e0d918bcSEd Tanous             // string at the beggining
36808777fb0SLewanczyk, Dawid             const std::string& sensorType = split[4];
36908777fb0SLewanczyk, Dawid             const std::string& sensorName = split[5];
37008777fb0SLewanczyk, Dawid             CROW_LOG_DEBUG << "sensorName " << sensorName << " sensorType "
37108777fb0SLewanczyk, Dawid                            << sensorType;
37208777fb0SLewanczyk, Dawid             if (sensorNames.find(sensorName) == sensorNames.end()) {
37308777fb0SLewanczyk, Dawid               CROW_LOG_ERROR << sensorName << " not in sensor list ";
37408777fb0SLewanczyk, Dawid               continue;
37508777fb0SLewanczyk, Dawid             }
37608777fb0SLewanczyk, Dawid 
37708777fb0SLewanczyk, Dawid             const char* fieldName = nullptr;
37808777fb0SLewanczyk, Dawid             if (sensorType == "temperature") {
37908777fb0SLewanczyk, Dawid               fieldName = "Temperatures";
38008777fb0SLewanczyk, Dawid             } else if (sensorType == "fan" || sensorType == "fan_tach") {
38108777fb0SLewanczyk, Dawid               fieldName = "Fans";
38208777fb0SLewanczyk, Dawid             } else if (sensorType == "voltage") {
38308777fb0SLewanczyk, Dawid               fieldName = "Voltages";
38408777fb0SLewanczyk, Dawid             } else if (sensorType == "current") {
38508777fb0SLewanczyk, Dawid               fieldName = "PowerSupply";
38608777fb0SLewanczyk, Dawid             } else if (sensorType == "power") {
38708777fb0SLewanczyk, Dawid               fieldName = "PowerSupply";
38808777fb0SLewanczyk, Dawid             } else {
38908777fb0SLewanczyk, Dawid               CROW_LOG_ERROR << "Unsure how to handle sensorType "
39008777fb0SLewanczyk, Dawid                              << sensorType;
39108777fb0SLewanczyk, Dawid               continue;
39208777fb0SLewanczyk, Dawid             }
39308777fb0SLewanczyk, Dawid 
394*e0d918bcSEd Tanous             nlohmann::json& temp_array = asyncResp->res.json_value[fieldName];
39508777fb0SLewanczyk, Dawid 
39608777fb0SLewanczyk, Dawid             // Create the array if it doesn't yet exist
39708777fb0SLewanczyk, Dawid             if (temp_array.is_array() == false) {
39808777fb0SLewanczyk, Dawid               temp_array = nlohmann::json::array();
39908777fb0SLewanczyk, Dawid             }
40008777fb0SLewanczyk, Dawid 
401*e0d918bcSEd Tanous             temp_array.push_back(
402*e0d918bcSEd Tanous                 {{"@odata.id", "/redfish/v1/Chassis/" + asyncResp->chassisId +
403*e0d918bcSEd Tanous                                    "/Thermal#/" + sensorName}});
40408777fb0SLewanczyk, Dawid             nlohmann::json& sensor_json = temp_array.back();
405*e0d918bcSEd Tanous             objectInterfacesToJson(sensorName, sensorType, objDictEntry.second,
406*e0d918bcSEd Tanous                                    sensor_json);
40708777fb0SLewanczyk, Dawid           }
40808777fb0SLewanczyk, Dawid         };
40908777fb0SLewanczyk, Dawid 
41008777fb0SLewanczyk, Dawid         crow::connections::system_bus->async_method_call(
411aa2e59c1SEd Tanous             getManagedObjectsCb, connection, "/xyz/openbmc_project/Sensors",
412aa2e59c1SEd Tanous             "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
41308777fb0SLewanczyk, Dawid       };
41408777fb0SLewanczyk, Dawid     };
41508777fb0SLewanczyk, Dawid     // Get connections and then pass it to get sensors
41608777fb0SLewanczyk, Dawid     getConnections(asyncResp, sensorNames, std::move(getConnectionCb));
41708777fb0SLewanczyk, Dawid   };
41808777fb0SLewanczyk, Dawid 
41908777fb0SLewanczyk, Dawid   // Get chassis information related to sensors
42008777fb0SLewanczyk, Dawid   getChassis(asyncResp, std::move(getChassisCb));
42108777fb0SLewanczyk, Dawid };
42208777fb0SLewanczyk, Dawid 
42308777fb0SLewanczyk, Dawid }  // namespace redfish
424