xref: /openbmc/bmcweb/features/redfish/lib/sensors.hpp (revision 03b5bae30e8e072b43e06c313d828552674f2a6e)
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 
27*03b5bae3SJames Feist 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 /**
41588c3f0dSKowalski, Kamil  * SensorsAsyncResp
4208777fb0SLewanczyk, Dawid  * Gathers data needed for response processing after async calls are done
4308777fb0SLewanczyk, Dawid  */
44588c3f0dSKowalski, Kamil class SensorsAsyncResp {
4508777fb0SLewanczyk, Dawid  public:
46588c3f0dSKowalski, Kamil   SensorsAsyncResp(crow::response& response, const std::string& chassisId,
4708777fb0SLewanczyk, Dawid                    const std::initializer_list<const char*> types)
48588c3f0dSKowalski, 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 
53588c3f0dSKowalski, 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   }
62588c3f0dSKowalski, 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;
68588c3f0dSKowalski, 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
74588c3f0dSKowalski, 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>
79588c3f0dSKowalski, Kamil void getConnections(std::shared_ptr<SensorsAsyncResp> SensorsAsyncResp,
8008777fb0SLewanczyk, Dawid                     const boost::container::flat_set<std::string>& sensorNames,
8108777fb0SLewanczyk, Dawid                     Callback&& callback) {
827885954aSLewanczyk, Dawid   CROW_LOG_DEBUG << "getConnections enter";
83*03b5bae3SJames Feist   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
88588c3f0dSKowalski, Kamil   auto resp_handler =
89588c3f0dSKowalski, Kamil       [ callback{std::move(callback)}, SensorsAsyncResp, sensorNames ](
9008777fb0SLewanczyk, Dawid           const boost::system::error_code ec, const GetSubTreeType& subtree) {
917885954aSLewanczyk, Dawid     CROW_LOG_DEBUG << "getConnections resp_handler enter";
92e0d918bcSEd Tanous     if (ec) {
93588c3f0dSKowalski, Kamil       SensorsAsyncResp->setErrorStatus();
947885954aSLewanczyk, Dawid       CROW_LOG_ERROR << "getConnections resp_handler: Dbus error " << ec;
9508777fb0SLewanczyk, Dawid       return;
9608777fb0SLewanczyk, Dawid     }
9708777fb0SLewanczyk, Dawid 
9808777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "Found " << subtree.size() << " subtrees";
9908777fb0SLewanczyk, Dawid 
10008777fb0SLewanczyk, Dawid     // Make unique list of connections only for requested sensor types and
10108777fb0SLewanczyk, Dawid     // found in the chassis
10208777fb0SLewanczyk, Dawid     boost::container::flat_set<std::string> connections;
10308777fb0SLewanczyk, Dawid     // Intrinsic to avoid malloc.  Most systems will have < 8 sensor producers
10408777fb0SLewanczyk, Dawid     connections.reserve(8);
10508777fb0SLewanczyk, Dawid 
1067885954aSLewanczyk, Dawid     CROW_LOG_DEBUG << "sensorNames list count: " << sensorNames.size();
10708777fb0SLewanczyk, Dawid     for (const std::string& tsensor : sensorNames) {
10808777fb0SLewanczyk, Dawid       CROW_LOG_DEBUG << "Sensor to find: " << tsensor;
10908777fb0SLewanczyk, Dawid     }
11008777fb0SLewanczyk, Dawid 
11108777fb0SLewanczyk, Dawid     for (const std::pair<
11208777fb0SLewanczyk, Dawid              std::string,
11308777fb0SLewanczyk, Dawid              std::vector<std::pair<std::string, std::vector<std::string>>>>&
11408777fb0SLewanczyk, Dawid              object : subtree) {
115588c3f0dSKowalski, Kamil       for (const char* type : SensorsAsyncResp->types) {
11608777fb0SLewanczyk, Dawid         if (boost::starts_with(object.first, type)) {
11708777fb0SLewanczyk, Dawid           auto lastPos = object.first.rfind('/');
11808777fb0SLewanczyk, Dawid           if (lastPos != std::string::npos) {
11908777fb0SLewanczyk, Dawid             std::string sensorName = object.first.substr(lastPos + 1);
12008777fb0SLewanczyk, Dawid 
12108777fb0SLewanczyk, Dawid             if (sensorNames.find(sensorName) != sensorNames.end()) {
12208777fb0SLewanczyk, Dawid               // For each connection name
12308777fb0SLewanczyk, Dawid               for (const std::pair<std::string, std::vector<std::string>>&
12408777fb0SLewanczyk, Dawid                        objData : object.second) {
1257885954aSLewanczyk, Dawid                 CROW_LOG_DEBUG << "Adding connection: " << objData.first;
12608777fb0SLewanczyk, Dawid                 connections.insert(objData.first);
12708777fb0SLewanczyk, Dawid               }
12808777fb0SLewanczyk, Dawid             }
12908777fb0SLewanczyk, Dawid           }
13008777fb0SLewanczyk, Dawid           break;
13108777fb0SLewanczyk, Dawid         }
13208777fb0SLewanczyk, Dawid       }
13308777fb0SLewanczyk, Dawid     }
13408777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "Found " << connections.size() << " connections";
13508777fb0SLewanczyk, Dawid     callback(std::move(connections));
1367885954aSLewanczyk, Dawid     CROW_LOG_DEBUG << "getConnections resp_handler exit";
13708777fb0SLewanczyk, Dawid   };
13808777fb0SLewanczyk, Dawid 
13908777fb0SLewanczyk, Dawid   // Make call to ObjectMapper to find all sensors objects
140aa2e59c1SEd Tanous   crow::connections::system_bus->async_method_call(
141daf36e2eSEd Tanous       std::move(resp_handler), "xyz.openbmc_project.ObjectMapper",
142aa2e59c1SEd Tanous       "/xyz/openbmc_project/object_mapper", "xyz.openbmc_project.ObjectMapper",
143aa2e59c1SEd Tanous       "GetSubTree", path, 2, interfaces);
1447885954aSLewanczyk, Dawid   CROW_LOG_DEBUG << "getConnections exit";
14508777fb0SLewanczyk, Dawid }
14608777fb0SLewanczyk, Dawid 
14708777fb0SLewanczyk, Dawid /**
14808777fb0SLewanczyk, Dawid  * @brief Retrieves requested chassis sensors and redundancy data from DBus .
149588c3f0dSKowalski, Kamil  * @param SensorsAsyncResp   Pointer to object holding response data
15008777fb0SLewanczyk, Dawid  * @param callback  Callback for next step in gathered sensor processing
15108777fb0SLewanczyk, Dawid  */
15208777fb0SLewanczyk, Dawid template <typename Callback>
153588c3f0dSKowalski, Kamil void getChassis(std::shared_ptr<SensorsAsyncResp> SensorsAsyncResp,
154588c3f0dSKowalski, Kamil                 Callback&& callback) {
1557885954aSLewanczyk, Dawid   CROW_LOG_DEBUG << "getChassis enter";
15608777fb0SLewanczyk, Dawid   // Process response from EntityManager and extract chassis data
157588c3f0dSKowalski, Kamil   auto resp_handler = [ callback{std::move(callback)}, SensorsAsyncResp ](
15808777fb0SLewanczyk, Dawid       const boost::system::error_code ec, ManagedObjectsVectorType& resp) {
1597885954aSLewanczyk, Dawid     CROW_LOG_DEBUG << "getChassis resp_handler enter";
16008777fb0SLewanczyk, Dawid     if (ec) {
1617885954aSLewanczyk, Dawid       CROW_LOG_ERROR << "getChassis resp_handler DBUS error: " << ec;
162588c3f0dSKowalski, Kamil       SensorsAsyncResp->setErrorStatus();
16308777fb0SLewanczyk, Dawid       return;
16408777fb0SLewanczyk, Dawid     }
16508777fb0SLewanczyk, Dawid     boost::container::flat_set<std::string> sensorNames;
16608777fb0SLewanczyk, Dawid 
167588c3f0dSKowalski, Kamil     //   SensorsAsyncResp->chassisId
168daf36e2eSEd Tanous     bool foundChassis = false;
169daf36e2eSEd Tanous     std::vector<std::string> split;
170daf36e2eSEd Tanous     // Reserve space for
171daf36e2eSEd Tanous     // /xyz/openbmc_project/inventory/<name>/<subname> + 3 subnames
172daf36e2eSEd Tanous     split.reserve(8);
173daf36e2eSEd Tanous 
174daf36e2eSEd Tanous     for (const auto& objDictEntry : resp) {
175daf36e2eSEd Tanous       const std::string& objectPath =
176daf36e2eSEd Tanous           static_cast<const std::string&>(objDictEntry.first);
177daf36e2eSEd Tanous       boost::algorithm::split(split, objectPath, boost::is_any_of("/"));
178daf36e2eSEd Tanous       if (split.size() < 2) {
179daf36e2eSEd Tanous         CROW_LOG_ERROR << "Got path that isn't long enough " << objectPath;
180daf36e2eSEd Tanous         split.clear();
181daf36e2eSEd Tanous         continue;
182daf36e2eSEd Tanous       }
183daf36e2eSEd Tanous       const std::string& sensorName = split.end()[-1];
184daf36e2eSEd Tanous       const std::string& chassisName = split.end()[-2];
185daf36e2eSEd Tanous 
186588c3f0dSKowalski, Kamil       if (chassisName != SensorsAsyncResp->chassisId) {
187daf36e2eSEd Tanous         split.clear();
188daf36e2eSEd Tanous         continue;
189daf36e2eSEd Tanous       }
1907885954aSLewanczyk, Dawid       CROW_LOG_DEBUG << "New sensor: " << sensorName;
191daf36e2eSEd Tanous       foundChassis = true;
19208777fb0SLewanczyk, Dawid       sensorNames.emplace(sensorName);
193daf36e2eSEd Tanous       split.clear();
19408777fb0SLewanczyk, Dawid     };
19508777fb0SLewanczyk, Dawid     CROW_LOG_DEBUG << "Found " << sensorNames.size() << " Sensor names";
19608777fb0SLewanczyk, Dawid 
19708777fb0SLewanczyk, Dawid     if (!foundChassis) {
198588c3f0dSKowalski, Kamil       CROW_LOG_INFO << "Unable to find chassis named "
199588c3f0dSKowalski, Kamil                     << SensorsAsyncResp->chassisId;
200588c3f0dSKowalski, Kamil       SensorsAsyncResp->res.result(boost::beast::http::status::not_found);
20108777fb0SLewanczyk, Dawid     } else {
20208777fb0SLewanczyk, Dawid       callback(sensorNames);
20308777fb0SLewanczyk, Dawid     }
2047885954aSLewanczyk, Dawid     CROW_LOG_DEBUG << "getChassis resp_handler exit";
20508777fb0SLewanczyk, Dawid   };
20608777fb0SLewanczyk, Dawid 
20708777fb0SLewanczyk, Dawid   // Make call to EntityManager to find all chassis objects
208aa2e59c1SEd Tanous   crow::connections::system_bus->async_method_call(
2097885954aSLewanczyk, Dawid       resp_handler, "xyz.openbmc_project.EntityManager", "/",
2107885954aSLewanczyk, Dawid       "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
2117885954aSLewanczyk, Dawid   CROW_LOG_DEBUG << "getChassis exit";
21208777fb0SLewanczyk, Dawid }
21308777fb0SLewanczyk, Dawid 
21408777fb0SLewanczyk, Dawid /**
21508777fb0SLewanczyk, Dawid  * @brief Builds a json sensor representation of a sensor.
21608777fb0SLewanczyk, Dawid  * @param sensorName  The name of the sensor to be built
217274fad5aSGunnar Mills  * @param sensorType  The type (temperature, fan_tach, etc) of the sensor to
21808777fb0SLewanczyk, Dawid  * build
21908777fb0SLewanczyk, Dawid  * @param interfacesDict  A dictionary of the interfaces and properties of said
22008777fb0SLewanczyk, Dawid  * interfaces to be built from
22108777fb0SLewanczyk, Dawid  * @param sensor_json  The json object to fill
22208777fb0SLewanczyk, Dawid  */
22308777fb0SLewanczyk, Dawid void objectInterfacesToJson(
22408777fb0SLewanczyk, Dawid     const std::string& sensorName, const std::string& sensorType,
22508777fb0SLewanczyk, Dawid     const boost::container::flat_map<
226aa2e59c1SEd Tanous         std::string, boost::container::flat_map<std::string, SensorVariant>>&
22708777fb0SLewanczyk, Dawid         interfacesDict,
22808777fb0SLewanczyk, Dawid     nlohmann::json& sensor_json) {
22908777fb0SLewanczyk, Dawid   // We need a value interface before we can do anything with it
23008777fb0SLewanczyk, Dawid   auto value_it = interfacesDict.find("xyz.openbmc_project.Sensor.Value");
23108777fb0SLewanczyk, Dawid   if (value_it == interfacesDict.end()) {
23208777fb0SLewanczyk, Dawid     CROW_LOG_ERROR << "Sensor doesn't have a value interface";
23308777fb0SLewanczyk, Dawid     return;
23408777fb0SLewanczyk, Dawid   }
23508777fb0SLewanczyk, Dawid 
23608777fb0SLewanczyk, Dawid   // Assume values exist as is (10^0 == 1) if no scale exists
23708777fb0SLewanczyk, Dawid   int64_t scaleMultiplier = 0;
23808777fb0SLewanczyk, Dawid 
23908777fb0SLewanczyk, Dawid   auto scale_it = value_it->second.find("Scale");
24008777fb0SLewanczyk, Dawid   // If a scale exists, pull value as int64, and use the scaling.
24108777fb0SLewanczyk, Dawid   if (scale_it != value_it->second.end()) {
242aa2e59c1SEd Tanous     const int64_t* int64Value =
243aa2e59c1SEd Tanous         mapbox::get_ptr<const int64_t>(scale_it->second);
24408777fb0SLewanczyk, Dawid     if (int64Value != nullptr) {
24508777fb0SLewanczyk, Dawid       scaleMultiplier = *int64Value;
24608777fb0SLewanczyk, Dawid     }
24708777fb0SLewanczyk, Dawid   }
24808777fb0SLewanczyk, Dawid 
24908777fb0SLewanczyk, Dawid   sensor_json["MemberId"] = sensorName;
25008777fb0SLewanczyk, Dawid   sensor_json["Name"] = sensorName;
25108777fb0SLewanczyk, Dawid   sensor_json["Status"]["State"] = "Enabled";
25208777fb0SLewanczyk, Dawid   sensor_json["Status"]["Health"] = "OK";
25308777fb0SLewanczyk, Dawid 
25408777fb0SLewanczyk, Dawid   // Parameter to set to override the type we get from dbus, and force it to
25508777fb0SLewanczyk, Dawid   // int, regardless of what is available.  This is used for schemas like fan,
25608777fb0SLewanczyk, Dawid   // that require integers, not floats.
25708777fb0SLewanczyk, Dawid   bool forceToInt = false;
25808777fb0SLewanczyk, Dawid 
25908777fb0SLewanczyk, Dawid   const char* unit = "Reading";
26008777fb0SLewanczyk, Dawid   if (sensorType == "temperature") {
26108777fb0SLewanczyk, Dawid     unit = "ReadingCelsius";
2627885954aSLewanczyk, Dawid     sensor_json["@odata.type"] = "#Thermal.v1_3_0.Temperature";
26308777fb0SLewanczyk, Dawid     // TODO(ed) Documentation says that path should be type fan_tach,
26408777fb0SLewanczyk, Dawid     // implementation seems to implement fan
265*03b5bae3SJames Feist   } else if (sensorType == "fan" || sensorType == "fan_tach") {
26608777fb0SLewanczyk, Dawid     unit = "Reading";
26708777fb0SLewanczyk, Dawid     sensor_json["ReadingUnits"] = "RPM";
2687885954aSLewanczyk, Dawid     sensor_json["@odata.type"] = "#Thermal.v1_3_0.Fan";
26908777fb0SLewanczyk, Dawid     forceToInt = true;
27008777fb0SLewanczyk, Dawid   } else if (sensorType == "voltage") {
27108777fb0SLewanczyk, Dawid     unit = "ReadingVolts";
2727885954aSLewanczyk, Dawid     sensor_json["@odata.type"] = "#Power.v1_0_0.Voltage";
27308777fb0SLewanczyk, Dawid   } else {
27408777fb0SLewanczyk, Dawid     CROW_LOG_ERROR << "Redfish cannot map object type for " << sensorName;
27508777fb0SLewanczyk, Dawid     return;
27608777fb0SLewanczyk, Dawid   }
27708777fb0SLewanczyk, Dawid   // Map of dbus interface name, dbus property name and redfish property_name
27808777fb0SLewanczyk, Dawid   std::vector<std::tuple<const char*, const char*, const char*>> properties;
27908777fb0SLewanczyk, Dawid   properties.reserve(7);
28008777fb0SLewanczyk, Dawid 
28108777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Value", "Value", unit);
28208777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Warning",
28308777fb0SLewanczyk, Dawid                           "WarningHigh", "UpperThresholdNonCritical");
28408777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Warning",
28508777fb0SLewanczyk, Dawid                           "WarningLow", "LowerThresholdNonCritical");
28608777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Critical",
28708777fb0SLewanczyk, Dawid                           "CriticalHigh", "UpperThresholdCritical");
28808777fb0SLewanczyk, Dawid   properties.emplace_back("xyz.openbmc_project.Sensor.Threshold.Critical",
28908777fb0SLewanczyk, Dawid                           "CriticalLow", "LowerThresholdCritical");
29008777fb0SLewanczyk, Dawid 
29108777fb0SLewanczyk, Dawid   if (sensorType == "temperature") {
29208777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MinValue",
29308777fb0SLewanczyk, Dawid                             "MinReadingRangeTemp");
29408777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MaxValue",
29508777fb0SLewanczyk, Dawid                             "MaxReadingRangeTemp");
29608777fb0SLewanczyk, Dawid   } else {
29708777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MinValue",
29808777fb0SLewanczyk, Dawid                             "MinReadingRange");
29908777fb0SLewanczyk, Dawid     properties.emplace_back("xyz.openbmc_project.Sensor.Value", "MaxValue",
30008777fb0SLewanczyk, Dawid                             "MaxReadingRange");
30108777fb0SLewanczyk, Dawid   }
30208777fb0SLewanczyk, Dawid 
30308777fb0SLewanczyk, Dawid   for (const std::tuple<const char*, const char*, const char*>& p :
30408777fb0SLewanczyk, Dawid        properties) {
30508777fb0SLewanczyk, Dawid     auto interfaceProperties = interfacesDict.find(std::get<0>(p));
30608777fb0SLewanczyk, Dawid     if (interfaceProperties != interfacesDict.end()) {
30708777fb0SLewanczyk, Dawid       auto value_it = interfaceProperties->second.find(std::get<1>(p));
30808777fb0SLewanczyk, Dawid       if (value_it != interfaceProperties->second.end()) {
309aa2e59c1SEd Tanous         const SensorVariant& valueVariant = value_it->second;
31008777fb0SLewanczyk, Dawid         nlohmann::json& value_it = sensor_json[std::get<2>(p)];
311aa2e59c1SEd Tanous 
31208777fb0SLewanczyk, Dawid         // Attempt to pull the int64 directly
313aa2e59c1SEd Tanous         const int64_t* int64Value =
314aa2e59c1SEd Tanous             mapbox::get_ptr<const int64_t>(valueVariant);
31508777fb0SLewanczyk, Dawid 
31608777fb0SLewanczyk, Dawid         if (int64Value != nullptr) {
31708777fb0SLewanczyk, Dawid           if (forceToInt || scaleMultiplier >= 0) {
31808777fb0SLewanczyk, Dawid             value_it = *int64Value * std::pow(10, scaleMultiplier);
31908777fb0SLewanczyk, Dawid           } else {
32008777fb0SLewanczyk, Dawid             value_it = *int64Value *
32108777fb0SLewanczyk, Dawid                        std::pow(10, static_cast<double>(scaleMultiplier));
32208777fb0SLewanczyk, Dawid           }
32308777fb0SLewanczyk, Dawid         }
32408777fb0SLewanczyk, Dawid         // Attempt to pull the float directly
325aa2e59c1SEd Tanous         const double* doubleValue = mapbox::get_ptr<const double>(valueVariant);
32608777fb0SLewanczyk, Dawid 
32708777fb0SLewanczyk, Dawid         if (doubleValue != nullptr) {
32808777fb0SLewanczyk, Dawid           if (!forceToInt) {
32908777fb0SLewanczyk, Dawid             value_it = *doubleValue *
33008777fb0SLewanczyk, Dawid                        std::pow(10, static_cast<double>(scaleMultiplier));
33108777fb0SLewanczyk, Dawid           } else {
33208777fb0SLewanczyk, Dawid             value_it = static_cast<int64_t>(*doubleValue *
33308777fb0SLewanczyk, Dawid                                             std::pow(10, scaleMultiplier));
33408777fb0SLewanczyk, Dawid           }
33508777fb0SLewanczyk, Dawid         }
33608777fb0SLewanczyk, Dawid       }
33708777fb0SLewanczyk, Dawid     }
33808777fb0SLewanczyk, Dawid   }
339e0d918bcSEd Tanous   CROW_LOG_DEBUG << "Added sensor " << sensorName;
34008777fb0SLewanczyk, Dawid }
34108777fb0SLewanczyk, Dawid 
34208777fb0SLewanczyk, Dawid /**
34308777fb0SLewanczyk, Dawid  * @brief Entry point for retrieving sensors data related to requested
34408777fb0SLewanczyk, Dawid  *        chassis.
345588c3f0dSKowalski, Kamil  * @param SensorsAsyncResp   Pointer to object holding response data
34608777fb0SLewanczyk, Dawid  */
347588c3f0dSKowalski, Kamil void getChassisData(std::shared_ptr<SensorsAsyncResp> SensorsAsyncResp) {
3487885954aSLewanczyk, Dawid   CROW_LOG_DEBUG << "getChassisData enter";
349588c3f0dSKowalski, Kamil   auto getChassisCb = [&, SensorsAsyncResp](
350588c3f0dSKowalski, Kamil                           boost::container::flat_set<std::string>&
35108777fb0SLewanczyk, Dawid                               sensorNames) {
3527885954aSLewanczyk, Dawid     CROW_LOG_DEBUG << "getChassisCb enter";
353588c3f0dSKowalski, Kamil     auto getConnectionCb =
354588c3f0dSKowalski, Kamil         [&, SensorsAsyncResp, sensorNames](
355588c3f0dSKowalski, Kamil             const boost::container::flat_set<std::string>& connections) {
3567885954aSLewanczyk, Dawid           CROW_LOG_DEBUG << "getConnectionCb enter";
35708777fb0SLewanczyk, Dawid           // Get managed objects from all services exposing sensors
35808777fb0SLewanczyk, Dawid           for (const std::string& connection : connections) {
35908777fb0SLewanczyk, Dawid             // Response handler to process managed objects
360588c3f0dSKowalski, Kamil             auto getManagedObjectsCb = [&, SensorsAsyncResp, sensorNames](
36108777fb0SLewanczyk, Dawid                                            const boost::system::error_code ec,
36208777fb0SLewanczyk, Dawid                                            ManagedObjectsVectorType& resp) {
3637885954aSLewanczyk, Dawid               CROW_LOG_DEBUG << "getManagedObjectsCb enter";
3647885954aSLewanczyk, Dawid               if (ec) {
3657885954aSLewanczyk, Dawid                 CROW_LOG_ERROR << "getManagedObjectsCb DBUS error: " << ec;
3667885954aSLewanczyk, Dawid                 SensorsAsyncResp->setErrorStatus();
3677885954aSLewanczyk, Dawid                 return;
3687885954aSLewanczyk, Dawid               }
36908777fb0SLewanczyk, Dawid               // Go through all objects and update response with
37008777fb0SLewanczyk, Dawid               // sensor data
37108777fb0SLewanczyk, Dawid               for (const auto& objDictEntry : resp) {
372aa2e59c1SEd Tanous                 const std::string& objPath =
373daf36e2eSEd Tanous                     static_cast<const std::string&>(objDictEntry.first);
374588c3f0dSKowalski, Kamil                 CROW_LOG_DEBUG << "getManagedObjectsCb parsing object "
375588c3f0dSKowalski, Kamil                                << objPath;
376e0d918bcSEd Tanous 
37708777fb0SLewanczyk, Dawid                 std::vector<std::string> split;
37808777fb0SLewanczyk, Dawid                 // Reserve space for
379*03b5bae3SJames Feist                 // /xyz/openbmc_project/sensors/<name>/<subname>
38008777fb0SLewanczyk, Dawid                 split.reserve(6);
38108777fb0SLewanczyk, Dawid                 boost::algorithm::split(split, objPath, boost::is_any_of("/"));
38208777fb0SLewanczyk, Dawid                 if (split.size() < 6) {
383588c3f0dSKowalski, Kamil                   CROW_LOG_ERROR << "Got path that isn't long enough "
384588c3f0dSKowalski, Kamil                                  << objPath;
38508777fb0SLewanczyk, Dawid                   continue;
38608777fb0SLewanczyk, Dawid                 }
38708777fb0SLewanczyk, Dawid                 // These indexes aren't intuitive, as boost::split puts an empty
388e0d918bcSEd Tanous                 // string at the beggining
38908777fb0SLewanczyk, Dawid                 const std::string& sensorType = split[4];
39008777fb0SLewanczyk, Dawid                 const std::string& sensorName = split[5];
39108777fb0SLewanczyk, Dawid                 CROW_LOG_DEBUG << "sensorName " << sensorName << " sensorType "
39208777fb0SLewanczyk, Dawid                                << sensorType;
39308777fb0SLewanczyk, Dawid                 if (sensorNames.find(sensorName) == sensorNames.end()) {
39408777fb0SLewanczyk, Dawid                   CROW_LOG_ERROR << sensorName << " not in sensor list ";
39508777fb0SLewanczyk, Dawid                   continue;
39608777fb0SLewanczyk, Dawid                 }
39708777fb0SLewanczyk, Dawid 
39808777fb0SLewanczyk, Dawid                 const char* fieldName = nullptr;
39908777fb0SLewanczyk, Dawid                 if (sensorType == "temperature") {
40008777fb0SLewanczyk, Dawid                   fieldName = "Temperatures";
40108777fb0SLewanczyk, Dawid                 } else if (sensorType == "fan" || sensorType == "fan_tach") {
40208777fb0SLewanczyk, Dawid                   fieldName = "Fans";
40308777fb0SLewanczyk, Dawid                 } else if (sensorType == "voltage") {
40408777fb0SLewanczyk, Dawid                   fieldName = "Voltages";
40508777fb0SLewanczyk, Dawid                 } else if (sensorType == "current") {
40608777fb0SLewanczyk, Dawid                   fieldName = "PowerSupply";
40708777fb0SLewanczyk, Dawid                 } else if (sensorType == "power") {
40808777fb0SLewanczyk, Dawid                   fieldName = "PowerSupply";
40908777fb0SLewanczyk, Dawid                 } else {
41008777fb0SLewanczyk, Dawid                   CROW_LOG_ERROR << "Unsure how to handle sensorType "
41108777fb0SLewanczyk, Dawid                                  << sensorType;
41208777fb0SLewanczyk, Dawid                   continue;
41308777fb0SLewanczyk, Dawid                 }
41408777fb0SLewanczyk, Dawid 
415588c3f0dSKowalski, Kamil                 nlohmann::json& temp_array =
416588c3f0dSKowalski, Kamil                     SensorsAsyncResp->res.json_value[fieldName];
41708777fb0SLewanczyk, Dawid 
41808777fb0SLewanczyk, Dawid                 // Create the array if it doesn't yet exist
41908777fb0SLewanczyk, Dawid                 if (temp_array.is_array() == false) {
42008777fb0SLewanczyk, Dawid                   temp_array = nlohmann::json::array();
42108777fb0SLewanczyk, Dawid                 }
42208777fb0SLewanczyk, Dawid 
423e0d918bcSEd Tanous                 temp_array.push_back(
424588c3f0dSKowalski, Kamil                     {{"@odata.id", "/redfish/v1/Chassis/" +
425588c3f0dSKowalski, Kamil                                        SensorsAsyncResp->chassisId +
426e0d918bcSEd Tanous                                        "/Thermal#/" + sensorName}});
42708777fb0SLewanczyk, Dawid                 nlohmann::json& sensor_json = temp_array.back();
428588c3f0dSKowalski, Kamil                 objectInterfacesToJson(sensorName, sensorType,
429588c3f0dSKowalski, Kamil                                        objDictEntry.second, sensor_json);
43008777fb0SLewanczyk, Dawid               }
4317885954aSLewanczyk, Dawid               CROW_LOG_DEBUG << "getManagedObjectsCb exit";
43208777fb0SLewanczyk, Dawid             };
43308777fb0SLewanczyk, Dawid             crow::connections::system_bus->async_method_call(
4347885954aSLewanczyk, Dawid                 getManagedObjectsCb, connection, "/",
435aa2e59c1SEd Tanous                 "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
43608777fb0SLewanczyk, Dawid           };
4377885954aSLewanczyk, Dawid           CROW_LOG_DEBUG << "getConnectionCb exit";
43808777fb0SLewanczyk, Dawid         };
43908777fb0SLewanczyk, Dawid     // Get connections and then pass it to get sensors
440588c3f0dSKowalski, Kamil     getConnections(SensorsAsyncResp, sensorNames, std::move(getConnectionCb));
4417885954aSLewanczyk, Dawid     CROW_LOG_DEBUG << "getChassisCb exit";
44208777fb0SLewanczyk, Dawid   };
44308777fb0SLewanczyk, Dawid 
44408777fb0SLewanczyk, Dawid   // Get chassis information related to sensors
445588c3f0dSKowalski, Kamil   getChassis(SensorsAsyncResp, std::move(getChassisCb));
4467885954aSLewanczyk, Dawid   CROW_LOG_DEBUG << "getChassisData exit";
44708777fb0SLewanczyk, Dawid };
44808777fb0SLewanczyk, Dawid 
44908777fb0SLewanczyk, Dawid }  // namespace redfish
450