#pragma once #include "app.hpp" #include "dbus_utility.hpp" #include "query.hpp" #include "registries/privilege_registry.hpp" #include "utils/collection.hpp" #include "utils/telemetry_utils.hpp" #include "utils/time_utils.hpp" #include #include #include #include namespace redfish { namespace telemetry { using Readings = std::vector>; using TimestampReadings = std::tuple; inline nlohmann::json toMetricValues(const Readings& readings) { nlohmann::json metricValues = nlohmann::json::array_t(); for (const auto& [metadata, sensorValue, timestamp] : readings) { nlohmann::json::object_t metricReport; metricReport["MetricProperty"] = metadata; metricReport["MetricValue"] = std::to_string(sensorValue); metricReport["Timestamp"] = redfish::time_utils::getDateTimeUintMs(timestamp); metricValues.emplace_back(std::move(metricReport)); } return metricValues; } inline bool fillReport(nlohmann::json& json, const std::string& id, const TimestampReadings& timestampReadings) { json["@odata.type"] = "#MetricReport.v1_3_0.MetricReport"; json["@odata.id"] = boost::urls::format( "/redfish/v1/TelemetryService/MetricReports/{}", id); json["Id"] = id; json["Name"] = id; json["MetricReportDefinition"]["@odata.id"] = boost::urls::format( "/redfish/v1/TelemetryService/MetricReportDefinitions/{}", id); const auto& [timestamp, readings] = timestampReadings; json["Timestamp"] = redfish::time_utils::getDateTimeUintMs(timestamp); json["MetricValues"] = toMetricValues(readings); return true; } } // namespace telemetry inline void requestRoutesMetricReportCollection(App& app) { BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReports/") .privileges(redfish::privileges::getMetricReportCollection) .methods(boost::beast::http::verb::get)( [&app](const crow::Request& req, const std::shared_ptr& asyncResp) { if (!redfish::setUpRedfishRoute(app, req, asyncResp)) { return; } asyncResp->res.jsonValue["@odata.type"] = "#MetricReportCollection.MetricReportCollection"; asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/TelemetryService/MetricReports"; asyncResp->res.jsonValue["Name"] = "Metric Report Collection"; constexpr std::array interfaces{ telemetry::reportInterface}; collection_util::getCollectionMembers( asyncResp, boost::urls::url("/redfish/v1/TelemetryService/MetricReports"), interfaces, "/xyz/openbmc_project/Telemetry/Reports/TelemetryService"); }); } inline void requestRoutesMetricReport(App& app) { BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReports//") .privileges(redfish::privileges::getMetricReport) .methods(boost::beast::http::verb::get)( [&app](const crow::Request& req, const std::shared_ptr& asyncResp, const std::string& id) { if (!redfish::setUpRedfishRoute(app, req, asyncResp)) { return; } const std::string reportPath = telemetry::getDbusReportPath(id); crow::connections::systemBus->async_method_call( [asyncResp, id, reportPath](const boost::system::error_code& ec) { if (ec.value() == EBADR || ec == boost::system::errc::host_unreachable) { messages::resourceNotFound(asyncResp->res, "MetricReport", id); return; } if (ec) { BMCWEB_LOG_ERROR("respHandler DBus error {}", ec); messages::internalError(asyncResp->res); return; } sdbusplus::asio::getProperty( *crow::connections::systemBus, telemetry::service, reportPath, telemetry::reportInterface, "Readings", [asyncResp, id](const boost::system::error_code& ec2, const telemetry::TimestampReadings& ret) { if (ec2) { BMCWEB_LOG_ERROR("respHandler DBus error {}", ec2); messages::internalError(asyncResp->res); return; } telemetry::fillReport(asyncResp->res.jsonValue, id, ret); }); }, telemetry::service, reportPath, telemetry::reportInterface, "Update"); }); } } // namespace redfish