xref: /openbmc/bmcweb/features/redfish/lib/sensors.hpp (revision 588c3f0da61ef4762e3ab2aa60e938fa6c8dce1c)
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 /**
41*588c3f0dSKowalski, Kamil  * SensorsAsyncResp
4208777fb0SLewanczyk, Dawid  * Gathers data needed for response processing after async calls are done
4308777fb0SLewanczyk, Dawid  */
44*588c3f0dSKowalski, Kamil class SensorsAsyncResp {
4508777fb0SLewanczyk, Dawid  public:
46*588c3f0dSKowalski, Kamil   SensorsAsyncResp(crow::response& response, const std::string& chassisId,
4708777fb0SLewanczyk, Dawid                    const std::initializer_list<const char*> types)
48*588c3f0dSKowalski, Kamil       : res(response), chassisId(chassisId), types(types) {
4908777fb0SLewanczyk, Dawid     res.json_value["@odata.id"] =
5008777fb0SLewanczyk, Dawid         "/redfish/v1/Chassis/" + chassisId + "/Thermal";
5108777fb0SLewanczyk, Dawid   }
5208777fb0SLewanczyk, Dawid 
53*588c3f0dSKowalski, Kamil   ~SensorsAsyncResp() {
54e0d918bcSEd 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   }
62*588c3f0dSKowalski, Kamil 
6308777fb0SLewanczyk, Dawid   void setErrorStatus() {
64e0d918bcSEd Tanous     res.result(boost::beast::http::status::internal_server_error);
6508777fb0SLewanczyk, Dawid   }
6608777fb0SLewanczyk, Dawid 
6708777fb0SLewanczyk, Dawid   crow::response& res;
68*588c3f0dSKowalski, Kamil   std::string chassisId{};
6908777fb0SLewanczyk, Dawid   const std::vector<const char*> types;
7008777fb0SLewanczyk, Dawid };
7108777fb0SLewanczyk, Dawid 
7208777fb0SLewanczyk, Dawid /**
7308777fb0SLewanczyk, Dawid  * @brief Creates connections necessary for chassis sensors
74*588c3f0dSKowalski, Kamil  * @param SensorsAsyncResp Pointer to object holding response data
7508777fb0SLewanczyk, Dawid  * @param sensorNames Sensors retrieved from chassis
7608777fb0SLewanczyk, Dawid  * @param callback Callback for processing gathered connections
7708777fb0SLewanczyk, Dawid  */
7808777fb0SLewanczyk, Dawid template <typename Callback>
79*588c3f0dSKowalski, Kamil void getConnections(std::shared_ptr<SensorsAsyncResp> SensorsAsyncResp,
8008777fb0SLewanczyk, Dawid                     const boost::container::flat_set<std::string>& sensorNames,
8108777fb0SLewanczyk, Dawid                     Callback&& callback) {
8208777fb0SLewanczyk, Dawid   CROW_LOG_DEBUG << "getConnections";
8308777fb0SLewanczyk, Dawid   const std::string path = "/xyz/openbmc_project/Sensors";
8408777fb0SLewanczyk, Dawid   const std::array<std::string, 1> interfaces = {
8508777fb0SLewanczyk, Dawid       "xyz.openbmc_project.Sensor.Value"};
8608777fb0SLewanczyk, Dawid 
8708777fb0SLewanczyk, Dawid   // Response handler for parsing objects subtree
88*588c3f0dSKowalski, Kamil   auto resp_handler =
89*588c3f0dSKowalski, Kamil       [ callback{std::move(callback)}, SensorsAsyncResp, sensorNames ](
9008777fb0SLewanczyk, Dawid           const boost::system::error_code ec, const GetSubTreeType& subtree) {
91e0d918bcSEd Tanous     if (ec) {
92*588c3f0dSKowalski, Kamil       SensorsAsyncResp->setErrorStatus();
93daf36e2eSEd Tanous       CROW_LOG_ERROR << "resp_handler: Dbus error " << ec;
9408777fb0SLewanczyk, Dawid       return;
9508777fb0SLewanczyk, Dawid     }
9608777fb0SLewanczyk, Dawid 
9708777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "Found " << subtree.size() << " subtrees";
9808777fb0SLewanczyk, Dawid 
9908777fb0SLewanczyk, Dawid     // Make unique list of connections only for requested sensor types and
10008777fb0SLewanczyk, Dawid     // found in the chassis
10108777fb0SLewanczyk, Dawid     boost::container::flat_set<std::string> connections;
10208777fb0SLewanczyk, Dawid     // Intrinsic to avoid malloc.  Most systems will have < 8 sensor producers
10308777fb0SLewanczyk, Dawid     connections.reserve(8);
10408777fb0SLewanczyk, Dawid 
10508777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "sensorNames list cout: " << sensorNames.size();
10608777fb0SLewanczyk, Dawid     for (const std::string& tsensor : sensorNames) {
10708777fb0SLewanczyk, Dawid       CROW_LOG_DEBUG << "Sensor to find: " << tsensor;
10808777fb0SLewanczyk, Dawid     }
10908777fb0SLewanczyk, Dawid 
11008777fb0SLewanczyk, Dawid     for (const std::pair<
11108777fb0SLewanczyk, Dawid              std::string,
11208777fb0SLewanczyk, Dawid              std::vector<std::pair<std::string, std::vector<std::string>>>>&
11308777fb0SLewanczyk, Dawid              object : subtree) {
114*588c3f0dSKowalski, Kamil       for (const char* type : SensorsAsyncResp->types) {
11508777fb0SLewanczyk, Dawid         if (boost::starts_with(object.first, type)) {
11608777fb0SLewanczyk, Dawid           auto lastPos = object.first.rfind('/');
11708777fb0SLewanczyk, Dawid           if (lastPos != std::string::npos) {
11808777fb0SLewanczyk, Dawid             std::string sensorName = object.first.substr(lastPos + 1);
11908777fb0SLewanczyk, Dawid 
12008777fb0SLewanczyk, Dawid             if (sensorNames.find(sensorName) != sensorNames.end()) {
12108777fb0SLewanczyk, Dawid               // For each connection name
12208777fb0SLewanczyk, Dawid               for (const std::pair<std::string, std::vector<std::string>>&
12308777fb0SLewanczyk, Dawid                        objData : object.second) {
12408777fb0SLewanczyk, Dawid                 connections.insert(objData.first);
12508777fb0SLewanczyk, Dawid               }
12608777fb0SLewanczyk, Dawid             }
12708777fb0SLewanczyk, Dawid           }
12808777fb0SLewanczyk, Dawid           break;
12908777fb0SLewanczyk, Dawid         }
13008777fb0SLewanczyk, Dawid       }
13108777fb0SLewanczyk, Dawid     }
13208777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "Found " << connections.size() << " connections";
13308777fb0SLewanczyk, Dawid     callback(std::move(connections));
13408777fb0SLewanczyk, Dawid   };
13508777fb0SLewanczyk, Dawid 
13608777fb0SLewanczyk, Dawid   // Make call to ObjectMapper to find all sensors objects
137aa2e59c1SEd Tanous   crow::connections::system_bus->async_method_call(
138daf36e2eSEd Tanous       std::move(resp_handler), "xyz.openbmc_project.ObjectMapper",
139aa2e59c1SEd Tanous       "/xyz/openbmc_project/object_mapper", "xyz.openbmc_project.ObjectMapper",
140aa2e59c1SEd Tanous       "GetSubTree", path, 2, interfaces);
14108777fb0SLewanczyk, Dawid }
14208777fb0SLewanczyk, Dawid 
14308777fb0SLewanczyk, Dawid /**
14408777fb0SLewanczyk, Dawid  * @brief Retrieves requested chassis sensors and redundancy data from DBus .
145*588c3f0dSKowalski, Kamil  * @param SensorsAsyncResp   Pointer to object holding response data
14608777fb0SLewanczyk, Dawid  * @param callback  Callback for next step in gathered sensor processing
14708777fb0SLewanczyk, Dawid  */
14808777fb0SLewanczyk, Dawid template <typename Callback>
149*588c3f0dSKowalski, Kamil void getChassis(std::shared_ptr<SensorsAsyncResp> SensorsAsyncResp,
150*588c3f0dSKowalski, Kamil                 Callback&& callback) {
15108777fb0SLewanczyk, Dawid   CROW_LOG_DEBUG << "getChassis Done";
15208777fb0SLewanczyk, Dawid 
15308777fb0SLewanczyk, Dawid   // Process response from EntityManager and extract chassis data
154*588c3f0dSKowalski, Kamil   auto resp_handler = [ callback{std::move(callback)}, SensorsAsyncResp ](
15508777fb0SLewanczyk, Dawid       const boost::system::error_code ec, ManagedObjectsVectorType& resp) {
15608777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "getChassis resp_handler called back Done";
15708777fb0SLewanczyk, Dawid     if (ec) {
15808777fb0SLewanczyk, Dawid       CROW_LOG_ERROR << "getChassis resp_handler got error " << ec;
159*588c3f0dSKowalski, Kamil       SensorsAsyncResp->setErrorStatus();
16008777fb0SLewanczyk, Dawid       return;
16108777fb0SLewanczyk, Dawid     }
16208777fb0SLewanczyk, Dawid     boost::container::flat_set<std::string> sensorNames;
16308777fb0SLewanczyk, Dawid 
164*588c3f0dSKowalski, Kamil     //   SensorsAsyncResp->chassisId
165daf36e2eSEd Tanous     bool foundChassis = false;
166daf36e2eSEd Tanous     std::vector<std::string> split;
167daf36e2eSEd Tanous     // Reserve space for
168daf36e2eSEd Tanous     // /xyz/openbmc_project/inventory/<name>/<subname> + 3 subnames
169daf36e2eSEd Tanous     split.reserve(8);
170daf36e2eSEd Tanous 
171daf36e2eSEd Tanous     for (const auto& objDictEntry : resp) {
172daf36e2eSEd Tanous       const std::string& objectPath =
173daf36e2eSEd Tanous           static_cast<const std::string&>(objDictEntry.first);
174daf36e2eSEd Tanous       boost::algorithm::split(split, objectPath, boost::is_any_of("/"));
175daf36e2eSEd Tanous       if (split.size() < 2) {
176daf36e2eSEd Tanous         CROW_LOG_ERROR << "Got path that isn't long enough " << objectPath;
177daf36e2eSEd Tanous         split.clear();
178daf36e2eSEd Tanous         continue;
179daf36e2eSEd Tanous       }
180daf36e2eSEd Tanous       const std::string& sensorName = split.end()[-1];
181daf36e2eSEd Tanous       const std::string& chassisName = split.end()[-2];
182daf36e2eSEd Tanous 
183*588c3f0dSKowalski, Kamil       if (chassisName != SensorsAsyncResp->chassisId) {
184daf36e2eSEd Tanous         split.clear();
185daf36e2eSEd Tanous         continue;
186daf36e2eSEd Tanous       }
187daf36e2eSEd Tanous       foundChassis = true;
18808777fb0SLewanczyk, Dawid       sensorNames.emplace(sensorName);
189daf36e2eSEd Tanous       split.clear();
19008777fb0SLewanczyk, Dawid     };
19108777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "Found " << sensorNames.size() << " Sensor names";
19208777fb0SLewanczyk, Dawid 
19308777fb0SLewanczyk, Dawid     if (!foundChassis) {
194*588c3f0dSKowalski, Kamil       CROW_LOG_INFO << "Unable to find chassis named "
195*588c3f0dSKowalski, Kamil                     << SensorsAsyncResp->chassisId;
196*588c3f0dSKowalski, Kamil       SensorsAsyncResp->res.result(boost::beast::http::status::not_found);
19708777fb0SLewanczyk, Dawid     } else {
19808777fb0SLewanczyk, Dawid       callback(sensorNames);
19908777fb0SLewanczyk, Dawid     }
20008777fb0SLewanczyk, Dawid   };
20108777fb0SLewanczyk, Dawid 
20208777fb0SLewanczyk, Dawid   // Make call to EntityManager to find all chassis objects
203aa2e59c1SEd Tanous   crow::connections::system_bus->async_method_call(
204aa2e59c1SEd Tanous       resp_handler, "xyz.openbmc_project.EntityManager",
205daf36e2eSEd Tanous       "/xyz/openbmc_project/inventory", "org.freedesktop.DBus.ObjectManager",
206daf36e2eSEd Tanous       "GetManagedObjects");
20708777fb0SLewanczyk, Dawid }
20808777fb0SLewanczyk, Dawid 
20908777fb0SLewanczyk, Dawid /**
21008777fb0SLewanczyk, Dawid  * @brief Builds a json sensor representation of a sensor.
21108777fb0SLewanczyk, Dawid  * @param sensorName  The name of the sensor to be built
212274fad5aSGunnar Mills  * @param sensorType  The type (temperature, fan_tach, etc) of the sensor to
21308777fb0SLewanczyk, Dawid  * build
21408777fb0SLewanczyk, Dawid  * @param interfacesDict  A dictionary of the interfaces and properties of said
21508777fb0SLewanczyk, Dawid  * interfaces to be built from
21608777fb0SLewanczyk, Dawid  * @param sensor_json  The json object to fill
21708777fb0SLewanczyk, Dawid  */
21808777fb0SLewanczyk, Dawid void objectInterfacesToJson(
21908777fb0SLewanczyk, Dawid     const std::string& sensorName, const std::string& sensorType,
22008777fb0SLewanczyk, Dawid     const boost::container::flat_map<
221aa2e59c1SEd Tanous         std::string, boost::container::flat_map<std::string, SensorVariant>>&
22208777fb0SLewanczyk, Dawid         interfacesDict,
22308777fb0SLewanczyk, Dawid     nlohmann::json& sensor_json) {
22408777fb0SLewanczyk, Dawid   // We need a value interface before we can do anything with it
22508777fb0SLewanczyk, Dawid   auto value_it = interfacesDict.find("xyz.openbmc_project.Sensor.Value");
22608777fb0SLewanczyk, Dawid   if (value_it == interfacesDict.end()) {
22708777fb0SLewanczyk, Dawid     CROW_LOG_ERROR << "Sensor doesn't have a value interface";
22808777fb0SLewanczyk, Dawid     return;
22908777fb0SLewanczyk, Dawid   }
23008777fb0SLewanczyk, Dawid 
23108777fb0SLewanczyk, Dawid   // Assume values exist as is (10^0 == 1) if no scale exists
23208777fb0SLewanczyk, Dawid   int64_t scaleMultiplier = 0;
23308777fb0SLewanczyk, Dawid 
23408777fb0SLewanczyk, Dawid   auto scale_it = value_it->second.find("Scale");
23508777fb0SLewanczyk, Dawid   // If a scale exists, pull value as int64, and use the scaling.
23608777fb0SLewanczyk, Dawid   if (scale_it != value_it->second.end()) {
237aa2e59c1SEd Tanous     const int64_t* int64Value =
238aa2e59c1SEd Tanous         mapbox::get_ptr<const int64_t>(scale_it->second);
23908777fb0SLewanczyk, Dawid     if (int64Value != nullptr) {
24008777fb0SLewanczyk, Dawid       scaleMultiplier = *int64Value;
24108777fb0SLewanczyk, Dawid     }
24208777fb0SLewanczyk, Dawid   }
24308777fb0SLewanczyk, Dawid 
24408777fb0SLewanczyk, Dawid   sensor_json["MemberId"] = sensorName;
24508777fb0SLewanczyk, Dawid   sensor_json["Name"] = sensorName;
24608777fb0SLewanczyk, Dawid   sensor_json["Status"]["State"] = "Enabled";
24708777fb0SLewanczyk, Dawid   sensor_json["Status"]["Health"] = "OK";
24808777fb0SLewanczyk, Dawid 
24908777fb0SLewanczyk, Dawid   // Parameter to set to override the type we get from dbus, and force it to
25008777fb0SLewanczyk, Dawid   // int, regardless of what is available.  This is used for schemas like fan,
25108777fb0SLewanczyk, Dawid   // that require integers, not floats.
25208777fb0SLewanczyk, Dawid   bool forceToInt = false;
25308777fb0SLewanczyk, Dawid 
25408777fb0SLewanczyk, Dawid   const char* unit = "Reading";
25508777fb0SLewanczyk, Dawid   if (sensorType == "temperature") {
25608777fb0SLewanczyk, Dawid     unit = "ReadingCelsius";
25708777fb0SLewanczyk, Dawid     // TODO(ed) Documentation says that path should be type fan_tach,
25808777fb0SLewanczyk, Dawid     // implementation seems to implement fan
259daf36e2eSEd Tanous   } else if (sensorType == "fan" || sensorType == "fan_type") {
26008777fb0SLewanczyk, Dawid     unit = "Reading";
26108777fb0SLewanczyk, Dawid     sensor_json["ReadingUnits"] = "RPM";
26208777fb0SLewanczyk, Dawid     forceToInt = true;
26308777fb0SLewanczyk, Dawid   } else if (sensorType == "voltage") {
26408777fb0SLewanczyk, Dawid     unit = "ReadingVolts";
26508777fb0SLewanczyk, Dawid   } else {
26608777fb0SLewanczyk, Dawid     CROW_LOG_ERROR << "Redfish cannot map object type for " << sensorName;
26708777fb0SLewanczyk, Dawid     return;
26808777fb0SLewanczyk, Dawid   }
26908777fb0SLewanczyk, Dawid   // Map of dbus interface name, dbus property name and redfish property_name
27008777fb0SLewanczyk, Dawid   std::vector<std::tuple<const char*, const char*, const char*>> properties;
27108777fb0SLewanczyk, Dawid   properties.reserve(7);
27208777fb0SLewanczyk, Dawid 
27308777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Value", "Value", unit);
27408777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Warning",
27508777fb0SLewanczyk, Dawid                           "WarningHigh", "UpperThresholdNonCritical");
27608777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Warning",
27708777fb0SLewanczyk, Dawid                           "WarningLow", "LowerThresholdNonCritical");
27808777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Critical",
27908777fb0SLewanczyk, Dawid                           "CriticalHigh", "UpperThresholdCritical");
28008777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Critical",
28108777fb0SLewanczyk, Dawid                           "CriticalLow", "LowerThresholdCritical");
28208777fb0SLewanczyk, Dawid 
28308777fb0SLewanczyk, Dawid   if (sensorType == "temperature") {
28408777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MinValue",
28508777fb0SLewanczyk, Dawid                             "MinReadingRangeTemp");
28608777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MaxValue",
28708777fb0SLewanczyk, Dawid                             "MaxReadingRangeTemp");
28808777fb0SLewanczyk, Dawid   } else {
28908777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MinValue",
29008777fb0SLewanczyk, Dawid                             "MinReadingRange");
29108777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MaxValue",
29208777fb0SLewanczyk, Dawid                             "MaxReadingRange");
29308777fb0SLewanczyk, Dawid   }
29408777fb0SLewanczyk, Dawid 
29508777fb0SLewanczyk, Dawid   for (const std::tuple<const char*, const char*, const char*>& p :
29608777fb0SLewanczyk, Dawid        properties) {
29708777fb0SLewanczyk, Dawid     auto interfaceProperties = interfacesDict.find(std::get<0>(p));
29808777fb0SLewanczyk, Dawid     if (interfaceProperties != interfacesDict.end()) {
29908777fb0SLewanczyk, Dawid       auto value_it = interfaceProperties->second.find(std::get<1>(p));
30008777fb0SLewanczyk, Dawid       if (value_it != interfaceProperties->second.end()) {
301aa2e59c1SEd Tanous         const SensorVariant& valueVariant = value_it->second;
30208777fb0SLewanczyk, Dawid         nlohmann::json& value_it = sensor_json[std::get<2>(p)];
303aa2e59c1SEd Tanous 
30408777fb0SLewanczyk, Dawid         // Attempt to pull the int64 directly
305aa2e59c1SEd Tanous         const int64_t* int64Value =
306aa2e59c1SEd Tanous             mapbox::get_ptr<const int64_t>(valueVariant);
30708777fb0SLewanczyk, Dawid 
30808777fb0SLewanczyk, Dawid         if (int64Value != nullptr) {
30908777fb0SLewanczyk, Dawid           if (forceToInt || scaleMultiplier >= 0) {
31008777fb0SLewanczyk, Dawid             value_it = *int64Value * std::pow(10, scaleMultiplier);
31108777fb0SLewanczyk, Dawid           } else {
31208777fb0SLewanczyk, Dawid             value_it = *int64Value *
31308777fb0SLewanczyk, Dawid                        std::pow(10, static_cast<double>(scaleMultiplier));
31408777fb0SLewanczyk, Dawid           }
31508777fb0SLewanczyk, Dawid         }
31608777fb0SLewanczyk, Dawid         // Attempt to pull the float directly
317aa2e59c1SEd Tanous         const double* doubleValue = mapbox::get_ptr<const double>(valueVariant);
31808777fb0SLewanczyk, Dawid 
31908777fb0SLewanczyk, Dawid         if (doubleValue != nullptr) {
32008777fb0SLewanczyk, Dawid           if (!forceToInt) {
32108777fb0SLewanczyk, Dawid             value_it = *doubleValue *
32208777fb0SLewanczyk, Dawid                        std::pow(10, static_cast<double>(scaleMultiplier));
32308777fb0SLewanczyk, Dawid           } else {
32408777fb0SLewanczyk, Dawid             value_it = static_cast<int64_t>(*doubleValue *
32508777fb0SLewanczyk, Dawid                                             std::pow(10, scaleMultiplier));
32608777fb0SLewanczyk, Dawid           }
32708777fb0SLewanczyk, Dawid         }
32808777fb0SLewanczyk, Dawid       }
32908777fb0SLewanczyk, Dawid     }
33008777fb0SLewanczyk, Dawid   }
331e0d918bcSEd Tanous   CROW_LOG_DEBUG << "Added sensor " << sensorName;
33208777fb0SLewanczyk, Dawid }
33308777fb0SLewanczyk, Dawid 
33408777fb0SLewanczyk, Dawid /**
33508777fb0SLewanczyk, Dawid  * @brief Entry point for retrieving sensors data related to requested
33608777fb0SLewanczyk, Dawid  *        chassis.
337*588c3f0dSKowalski, Kamil  * @param SensorsAsyncResp   Pointer to object holding response data
33808777fb0SLewanczyk, Dawid  */
339*588c3f0dSKowalski, Kamil void getChassisData(std::shared_ptr<SensorsAsyncResp> SensorsAsyncResp) {
34008777fb0SLewanczyk, Dawid   CROW_LOG_DEBUG << "getChassisData";
341*588c3f0dSKowalski, Kamil   auto getChassisCb = [&, SensorsAsyncResp](
342*588c3f0dSKowalski, Kamil                           boost::container::flat_set<std::string>&
34308777fb0SLewanczyk, Dawid                               sensorNames) {
34408777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "getChassisCb Done";
345*588c3f0dSKowalski, Kamil     auto getConnectionCb =
346*588c3f0dSKowalski, Kamil         [&, SensorsAsyncResp, sensorNames](
347*588c3f0dSKowalski, Kamil             const boost::container::flat_set<std::string>& connections) {
34808777fb0SLewanczyk, Dawid           CROW_LOG_DEBUG << "getConnectionCb Done";
34908777fb0SLewanczyk, Dawid           // Get managed objects from all services exposing sensors
35008777fb0SLewanczyk, Dawid           for (const std::string& connection : connections) {
35108777fb0SLewanczyk, Dawid             // Response handler to process managed objects
352*588c3f0dSKowalski, Kamil             auto getManagedObjectsCb = [&, SensorsAsyncResp, sensorNames](
35308777fb0SLewanczyk, Dawid                                            const boost::system::error_code ec,
35408777fb0SLewanczyk, Dawid                                            ManagedObjectsVectorType& resp) {
35508777fb0SLewanczyk, Dawid               // Go through all objects and update response with
35608777fb0SLewanczyk, Dawid               // sensor data
35708777fb0SLewanczyk, Dawid               for (const auto& objDictEntry : resp) {
358aa2e59c1SEd Tanous                 const std::string& objPath =
359daf36e2eSEd Tanous                     static_cast<const std::string&>(objDictEntry.first);
360*588c3f0dSKowalski, Kamil                 CROW_LOG_DEBUG << "getManagedObjectsCb parsing object "
361*588c3f0dSKowalski, Kamil                                << objPath;
362e0d918bcSEd Tanous 
36308777fb0SLewanczyk, Dawid                 std::vector<std::string> split;
36408777fb0SLewanczyk, Dawid                 // Reserve space for
36508777fb0SLewanczyk, Dawid                 // /xyz/openbmc_project/Sensors/<name>/<subname>
36608777fb0SLewanczyk, Dawid                 split.reserve(6);
36708777fb0SLewanczyk, Dawid                 boost::algorithm::split(split, objPath, boost::is_any_of("/"));
36808777fb0SLewanczyk, Dawid                 if (split.size() < 6) {
369*588c3f0dSKowalski, Kamil                   CROW_LOG_ERROR << "Got path that isn't long enough "
370*588c3f0dSKowalski, Kamil                                  << objPath;
37108777fb0SLewanczyk, Dawid                   continue;
37208777fb0SLewanczyk, Dawid                 }
37308777fb0SLewanczyk, Dawid                 // These indexes aren't intuitive, as boost::split puts an empty
374e0d918bcSEd Tanous                 // string at the beggining
37508777fb0SLewanczyk, Dawid                 const std::string& sensorType = split[4];
37608777fb0SLewanczyk, Dawid                 const std::string& sensorName = split[5];
37708777fb0SLewanczyk, Dawid                 CROW_LOG_DEBUG << "sensorName " << sensorName << " sensorType "
37808777fb0SLewanczyk, Dawid                                << sensorType;
37908777fb0SLewanczyk, Dawid                 if (sensorNames.find(sensorName) == sensorNames.end()) {
38008777fb0SLewanczyk, Dawid                   CROW_LOG_ERROR << sensorName << " not in sensor list ";
38108777fb0SLewanczyk, Dawid                   continue;
38208777fb0SLewanczyk, Dawid                 }
38308777fb0SLewanczyk, Dawid 
38408777fb0SLewanczyk, Dawid                 const char* fieldName = nullptr;
38508777fb0SLewanczyk, Dawid                 if (sensorType == "temperature") {
38608777fb0SLewanczyk, Dawid                   fieldName = "Temperatures";
38708777fb0SLewanczyk, Dawid                 } else if (sensorType == "fan" || sensorType == "fan_tach") {
38808777fb0SLewanczyk, Dawid                   fieldName = "Fans";
38908777fb0SLewanczyk, Dawid                 } else if (sensorType == "voltage") {
39008777fb0SLewanczyk, Dawid                   fieldName = "Voltages";
39108777fb0SLewanczyk, Dawid                 } else if (sensorType == "current") {
39208777fb0SLewanczyk, Dawid                   fieldName = "PowerSupply";
39308777fb0SLewanczyk, Dawid                 } else if (sensorType == "power") {
39408777fb0SLewanczyk, Dawid                   fieldName = "PowerSupply";
39508777fb0SLewanczyk, Dawid                 } else {
39608777fb0SLewanczyk, Dawid                   CROW_LOG_ERROR << "Unsure how to handle sensorType "
39708777fb0SLewanczyk, Dawid                                  << sensorType;
39808777fb0SLewanczyk, Dawid                   continue;
39908777fb0SLewanczyk, Dawid                 }
40008777fb0SLewanczyk, Dawid 
401*588c3f0dSKowalski, Kamil                 nlohmann::json& temp_array =
402*588c3f0dSKowalski, Kamil                     SensorsAsyncResp->res.json_value[fieldName];
40308777fb0SLewanczyk, Dawid 
40408777fb0SLewanczyk, Dawid                 // Create the array if it doesn't yet exist
40508777fb0SLewanczyk, Dawid                 if (temp_array.is_array() == false) {
40608777fb0SLewanczyk, Dawid                   temp_array = nlohmann::json::array();
40708777fb0SLewanczyk, Dawid                 }
40808777fb0SLewanczyk, Dawid 
409e0d918bcSEd Tanous                 temp_array.push_back(
410*588c3f0dSKowalski, Kamil                     {{"@odata.id", "/redfish/v1/Chassis/" +
411*588c3f0dSKowalski, Kamil                                        SensorsAsyncResp->chassisId +
412e0d918bcSEd Tanous                                        "/Thermal#/" + sensorName}});
41308777fb0SLewanczyk, Dawid                 nlohmann::json& sensor_json = temp_array.back();
414*588c3f0dSKowalski, Kamil                 objectInterfacesToJson(sensorName, sensorType,
415*588c3f0dSKowalski, Kamil                                        objDictEntry.second, sensor_json);
41608777fb0SLewanczyk, Dawid               }
41708777fb0SLewanczyk, Dawid             };
41808777fb0SLewanczyk, Dawid 
41908777fb0SLewanczyk, Dawid             crow::connections::system_bus->async_method_call(
420aa2e59c1SEd Tanous                 getManagedObjectsCb, connection, "/xyz/openbmc_project/Sensors",
421aa2e59c1SEd Tanous                 "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
42208777fb0SLewanczyk, Dawid           };
42308777fb0SLewanczyk, Dawid         };
42408777fb0SLewanczyk, Dawid     // Get connections and then pass it to get sensors
425*588c3f0dSKowalski, Kamil     getConnections(SensorsAsyncResp, sensorNames, std::move(getConnectionCb));
42608777fb0SLewanczyk, Dawid   };
42708777fb0SLewanczyk, Dawid 
42808777fb0SLewanczyk, Dawid   // Get chassis information related to sensors
429*588c3f0dSKowalski, Kamil   getChassis(SensorsAsyncResp, std::move(getChassisCb));
43008777fb0SLewanczyk, Dawid };
43108777fb0SLewanczyk, Dawid 
43208777fb0SLewanczyk, Dawid }  // namespace redfish
433