#pragma once #include "app.hpp" #include "dbus_utility.hpp" #include "generated/enums/metric_report_definition.hpp" #include "query.hpp" #include "registries/privilege_registry.hpp" #include "sensors.hpp" #include "utils/collection.hpp" #include "utils/dbus_utils.hpp" #include "utils/telemetry_utils.hpp" #include "utils/time_utils.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace redfish { namespace telemetry { using ReadingParameters = std::vector>, std::string, std::string, uint64_t>>; inline metric_report_definition::ReportActionsEnum toRedfishReportAction(std::string_view dbusValue) { if (dbusValue == "xyz.openbmc_project.Telemetry.Report.ReportActions.EmitsReadingsUpdate") { return metric_report_definition::ReportActionsEnum::RedfishEvent; } if (dbusValue == "xyz.openbmc_project.Telemetry.Report.ReportActions.LogToMetricReportsCollection") { return metric_report_definition::ReportActionsEnum:: LogToMetricReportsCollection; } return metric_report_definition::ReportActionsEnum::Invalid; } inline std::string toDbusReportAction(std::string_view redfishValue) { if (redfishValue == "RedfishEvent") { return "xyz.openbmc_project.Telemetry.Report.ReportActions.EmitsReadingsUpdate"; } if (redfishValue == "LogToMetricReportsCollection") { return "xyz.openbmc_project.Telemetry.Report.ReportActions.LogToMetricReportsCollection"; } return ""; } inline metric_report_definition::MetricReportDefinitionType toRedfishReportingType(std::string_view dbusValue) { if (dbusValue == "xyz.openbmc_project.Telemetry.Report.ReportingType.OnChange") { return metric_report_definition::MetricReportDefinitionType::OnChange; } if (dbusValue == "xyz.openbmc_project.Telemetry.Report.ReportingType.OnRequest") { return metric_report_definition::MetricReportDefinitionType::OnRequest; } if (dbusValue == "xyz.openbmc_project.Telemetry.Report.ReportingType.Periodic") { return metric_report_definition::MetricReportDefinitionType::Periodic; } return metric_report_definition::MetricReportDefinitionType::Invalid; } inline std::string toDbusReportingType(std::string_view redfishValue) { if (redfishValue == "OnChange") { return "xyz.openbmc_project.Telemetry.Report.ReportingType.OnChange"; } if (redfishValue == "OnRequest") { return "xyz.openbmc_project.Telemetry.Report.ReportingType.OnRequest"; } if (redfishValue == "Periodic") { return "xyz.openbmc_project.Telemetry.Report.ReportingType.Periodic"; } return ""; } inline metric_report_definition::CollectionTimeScope toRedfishCollectionTimeScope(std::string_view dbusValue) { if (dbusValue == "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.Point") { return metric_report_definition::CollectionTimeScope::Point; } if (dbusValue == "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.Interval") { return metric_report_definition::CollectionTimeScope::Interval; } if (dbusValue == "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.StartupInterval") { return metric_report_definition::CollectionTimeScope::StartupInterval; } return metric_report_definition::CollectionTimeScope::Invalid; } inline std::string toDbusCollectionTimeScope(std::string_view redfishValue) { if (redfishValue == "Point") { return "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.Point"; } if (redfishValue == "Interval") { return "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.Interval"; } if (redfishValue == "StartupInterval") { return "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.StartupInterval"; } return ""; } inline metric_report_definition::ReportUpdatesEnum toRedfishReportUpdates(std::string_view dbusValue) { if (dbusValue == "xyz.openbmc_project.Telemetry.Report.ReportUpdates.Overwrite") { return metric_report_definition::ReportUpdatesEnum::Overwrite; } if (dbusValue == "xyz.openbmc_project.Telemetry.Report.ReportUpdates.AppendWrapsWhenFull") { return metric_report_definition::ReportUpdatesEnum::AppendWrapsWhenFull; } if (dbusValue == "xyz.openbmc_project.Telemetry.Report.ReportUpdates.AppendStopsWhenFull") { return metric_report_definition::ReportUpdatesEnum::AppendStopsWhenFull; } return metric_report_definition::ReportUpdatesEnum::Invalid; } inline std::string toDbusReportUpdates(std::string_view redfishValue) { if (redfishValue == "Overwrite") { return "xyz.openbmc_project.Telemetry.Report.ReportUpdates.Overwrite"; } if (redfishValue == "AppendWrapsWhenFull") { return "xyz.openbmc_project.Telemetry.Report.ReportUpdates.AppendWrapsWhenFull"; } if (redfishValue == "AppendStopsWhenFull") { return "xyz.openbmc_project.Telemetry.Report.ReportUpdates.AppendStopsWhenFull"; } return ""; } inline std::optional getLinkedTriggers( std::span triggerPaths) { nlohmann::json::array_t triggers; for (const sdbusplus::message::object_path& path : triggerPaths) { if (path.parent_path() != "/xyz/openbmc_project/Telemetry/Triggers/TelemetryService") { BMCWEB_LOG_ERROR << "Property Triggers contains invalid value: " << path.str; return std::nullopt; } std::string id = path.filename(); if (id.empty()) { BMCWEB_LOG_ERROR << "Property Triggers contains invalid value: " << path.str; return std::nullopt; } nlohmann::json::object_t trigger; trigger["@odata.id"] = boost::urls::format("/redfish/v1/TelemetryService/Triggers/{}", id); triggers.emplace_back(std::move(trigger)); } return triggers; } inline void fillReportDefinition(const std::shared_ptr& asyncResp, const std::string& id, const dbus::utility::DBusPropertiesMap& properties) { std::vector reportActions; ReadingParameters readingParams; std::string reportingType; std::string reportUpdates; std::string name; uint64_t appendLimit = 0; uint64_t interval = 0; bool enabled = false; std::vector triggers; const bool success = sdbusplus::unpackPropertiesNoThrow( dbus_utils::UnpackErrorPrinter(), properties, "ReportingType", reportingType, "Interval", interval, "ReportActions", reportActions, "ReportUpdates", reportUpdates, "AppendLimit", appendLimit, "ReadingParameters", readingParams, "Name", name, "Enabled", enabled, "Triggers", triggers); if (!success) { messages::internalError(asyncResp->res); return; } metric_report_definition::MetricReportDefinitionType redfishReportingType = toRedfishReportingType(reportingType); if (redfishReportingType == metric_report_definition::MetricReportDefinitionType::Invalid) { messages::internalError(asyncResp->res); return; } asyncResp->res.jsonValue["MetricReportDefinitionType"] = redfishReportingType; std::optional linkedTriggers = getLinkedTriggers(triggers); if (!linkedTriggers) { messages::internalError(asyncResp->res); return; } asyncResp->res.jsonValue["Links"]["Triggers"] = std::move(*linkedTriggers); nlohmann::json::array_t redfishReportActions; for (const std::string& action : reportActions) { metric_report_definition::ReportActionsEnum redfishAction = toRedfishReportAction(action); if (redfishAction == metric_report_definition::ReportActionsEnum::Invalid) { messages::internalError(asyncResp->res); return; } redfishReportActions.emplace_back(redfishAction); } asyncResp->res.jsonValue["ReportActions"] = std::move(redfishReportActions); nlohmann::json::array_t metrics = nlohmann::json::array(); for (const auto& [sensorData, collectionFunction, collectionTimeScope, collectionDuration] : readingParams) { nlohmann::json::array_t metricProperties; for (const auto& [sensorPath, sensorMetadata] : sensorData) { metricProperties.emplace_back(sensorMetadata); } nlohmann::json::object_t metric; metric_report_definition::CalculationAlgorithmEnum redfishCollectionFunction = telemetry::toRedfishCollectionFunction(collectionFunction); if (redfishCollectionFunction == metric_report_definition::CalculationAlgorithmEnum::Invalid) { messages::internalError(asyncResp->res); return; } metric["CollectionFunction"] = redfishCollectionFunction; metric_report_definition::CollectionTimeScope redfishCollectionTimeScope = toRedfishCollectionTimeScope(collectionTimeScope); if (redfishCollectionTimeScope == metric_report_definition::CollectionTimeScope::Invalid) { messages::internalError(asyncResp->res); return; } metric["CollectionTimeScope"] = redfishCollectionTimeScope; metric["MetricProperties"] = std::move(metricProperties); metric["CollectionDuration"] = time_utils::toDurationString( std::chrono::milliseconds(collectionDuration)); metrics.emplace_back(std::move(metric)); } asyncResp->res.jsonValue["Metrics"] = std::move(metrics); if (enabled) { asyncResp->res.jsonValue["Status"]["State"] = "Enabled"; } else { asyncResp->res.jsonValue["Status"]["State"] = "Disabled"; } metric_report_definition::ReportUpdatesEnum redfishReportUpdates = toRedfishReportUpdates(reportUpdates); if (redfishReportUpdates == metric_report_definition::ReportUpdatesEnum::Invalid) { messages::internalError(asyncResp->res); return; } asyncResp->res.jsonValue["ReportUpdates"] = redfishReportUpdates; asyncResp->res.jsonValue["MetricReportDefinitionEnabled"] = enabled; asyncResp->res.jsonValue["AppendLimit"] = appendLimit; asyncResp->res.jsonValue["Name"] = name; asyncResp->res.jsonValue["Schedule"]["RecurrenceInterval"] = time_utils::toDurationString(std::chrono::milliseconds(interval)); asyncResp->res.jsonValue["@odata.type"] = "#MetricReportDefinition.v1_3_0.MetricReportDefinition"; asyncResp->res.jsonValue["@odata.id"] = boost::urls::format( "/redfish/v1/TelemetryService/MetricReportDefinitions/{}", id); asyncResp->res.jsonValue["Id"] = id; asyncResp->res.jsonValue["MetricReport"]["@odata.id"] = boost::urls::format( "/redfish/v1/TelemetryService/MetricReports/{}", id); } struct AddReportArgs { struct MetricArgs { std::vector uris; std::string collectionFunction; std::string collectionTimeScope; uint64_t collectionDuration = 0; }; std::string id; std::string name; std::string reportingType; std::string reportUpdates; uint64_t appendLimit = std::numeric_limits::max(); std::vector reportActions; uint64_t interval = std::numeric_limits::max(); std::vector metrics; bool metricReportDefinitionEnabled = true; }; inline bool toDbusReportActions(crow::Response& res, const std::vector& actions, AddReportArgs& args) { size_t index = 0; for (const std::string& action : actions) { std::string dbusReportAction = toDbusReportAction(action); if (dbusReportAction.empty()) { messages::propertyValueNotInList(res, nlohmann::json(action).dump(), "ReportActions/" + std::to_string(index)); return false; } args.reportActions.emplace_back(std::move(dbusReportAction)); index++; } return true; } inline bool getUserMetric(crow::Response& res, nlohmann::json& metric, AddReportArgs::MetricArgs& metricArgs) { std::optional> uris; std::optional collectionDurationStr; std::optional collectionFunction; std::optional collectionTimeScopeStr; if (!json_util::readJson(metric, res, "MetricProperties", uris, "CollectionFunction", collectionFunction, "CollectionTimeScope", collectionTimeScopeStr, "CollectionDuration", collectionDurationStr)) { return false; } if (uris) { metricArgs.uris = std::move(*uris); } if (collectionFunction) { std::string dbusCollectionFunction = telemetry::toDbusCollectionFunction(*collectionFunction); if (dbusCollectionFunction.empty()) { messages::propertyValueIncorrect(res, "CollectionFunction", *collectionFunction); return false; } metricArgs.collectionFunction = std::move(dbusCollectionFunction); } if (collectionTimeScopeStr) { std::string dbusCollectionTimeScope = toDbusCollectionTimeScope(*collectionTimeScopeStr); if (dbusCollectionTimeScope.empty()) { messages::propertyValueIncorrect(res, "CollectionTimeScope", *collectionTimeScopeStr); return false; } metricArgs.collectionTimeScope = std::move(dbusCollectionTimeScope); } if (collectionDurationStr) { std::optional duration = time_utils::fromDurationString(*collectionDurationStr); if (!duration || duration->count() < 0) { messages::propertyValueIncorrect(res, "CollectionDuration", *collectionDurationStr); return false; } metricArgs.collectionDuration = static_cast(duration->count()); } return true; } inline bool getUserMetrics(crow::Response& res, std::span metrics, std::vector& result) { result.reserve(metrics.size()); for (nlohmann::json& m : metrics) { AddReportArgs::MetricArgs metricArgs; if (!getUserMetric(res, m, metricArgs)) { return false; } result.emplace_back(std::move(metricArgs)); } return true; } inline bool getUserParameters(crow::Response& res, const crow::Request& req, AddReportArgs& args) { std::optional id; std::optional name; std::optional reportingTypeStr; std::optional reportUpdatesStr; std::optional appendLimit; std::optional metricReportDefinitionEnabled; std::optional> metrics; std::optional> reportActionsStr; std::optional schedule; if (!json_util::readJsonPatch( req, res, "Id", id, "Name", name, "Metrics", metrics, "MetricReportDefinitionType", reportingTypeStr, "ReportUpdates", reportUpdatesStr, "AppendLimit", appendLimit, "ReportActions", reportActionsStr, "Schedule", schedule, "MetricReportDefinitionEnabled", metricReportDefinitionEnabled)) { return false; } if (id) { constexpr const char* allowedCharactersInId = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_"; if (id->empty() || id->find_first_not_of(allowedCharactersInId) != std::string::npos) { messages::propertyValueIncorrect(res, "Id", *id); return false; } args.id = *id; } if (name) { args.name = *name; } if (reportingTypeStr) { std::string dbusReportingType = toDbusReportingType(*reportingTypeStr); if (dbusReportingType.empty()) { messages::propertyValueNotInList(res, *reportingTypeStr, "MetricReportDefinitionType"); return false; } args.reportingType = dbusReportingType; } if (reportUpdatesStr) { std::string dbusReportUpdates = toDbusReportUpdates(*reportUpdatesStr); if (dbusReportUpdates.empty()) { messages::propertyValueNotInList(res, *reportUpdatesStr, "ReportUpdates"); return false; } args.reportUpdates = dbusReportUpdates; } if (appendLimit) { args.appendLimit = *appendLimit; } if (metricReportDefinitionEnabled) { args.metricReportDefinitionEnabled = *metricReportDefinitionEnabled; } if (reportActionsStr) { if (!toDbusReportActions(res, *reportActionsStr, args)) { return false; } } if (reportingTypeStr == "Periodic") { if (!schedule) { messages::createFailedMissingReqProperties(res, "Schedule"); return false; } std::string durationStr; if (!json_util::readJson(*schedule, res, "RecurrenceInterval", durationStr)) { return false; } std::optional durationNum = time_utils::fromDurationString(durationStr); if (!durationNum || durationNum->count() < 0) { messages::propertyValueIncorrect(res, "RecurrenceInterval", durationStr); return false; } args.interval = static_cast(durationNum->count()); } if (metrics) { if (!getUserMetrics(res, *metrics, args.metrics)) { return false; } } return true; } inline bool getChassisSensorNodeFromMetrics( const std::shared_ptr& asyncResp, std::span metrics, boost::container::flat_set>& matched) { for (const auto& metric : metrics) { std::optional error = getChassisSensorNode(metric.uris, matched); if (error) { messages::propertyValueIncorrect(asyncResp->res, error->uri, "MetricProperties/" + std::to_string(error->index)); return false; } } return true; } class AddReport { public: AddReport(AddReportArgs argsIn, const std::shared_ptr& asyncRespIn) : asyncResp(asyncRespIn), args{std::move(argsIn)} {} ~AddReport() { boost::asio::post(crow::connections::systemBus->get_io_context(), std::bind_front(&performAddReport, asyncResp, args, std::move(uriToDbus))); } static void performAddReport( const std::shared_ptr& asyncResp, const AddReportArgs& args, const boost::container::flat_map& uriToDbus) { if (asyncResp->res.result() != boost::beast::http::status::ok) { return; } telemetry::ReadingParameters readingParams; readingParams.reserve(args.metrics.size()); for (const auto& metric : args.metrics) { std::vector< std::tuple> sensorParams; sensorParams.reserve(metric.uris.size()); for (size_t i = 0; i < metric.uris.size(); i++) { const std::string& uri = metric.uris[i]; auto el = uriToDbus.find(uri); if (el == uriToDbus.end()) { BMCWEB_LOG_ERROR << "Failed to find DBus sensor corresponding to URI " << uri; messages::propertyValueNotInList(asyncResp->res, uri, "MetricProperties/" + std::to_string(i)); return; } const std::string& dbusPath = el->second; sensorParams.emplace_back(dbusPath, uri); } readingParams.emplace_back( std::move(sensorParams), metric.collectionFunction, metric.collectionTimeScope, metric.collectionDuration); } crow::connections::systemBus->async_method_call( [asyncResp, id = args.id, uriToDbus]( const boost::system::error_code& ec, const std::string&) { if (ec == boost::system::errc::file_exists) { messages::resourceAlreadyExists( asyncResp->res, "MetricReportDefinition", "Id", id); return; } if (ec == boost::system::errc::too_many_files_open) { messages::createLimitReachedForResource(asyncResp->res); return; } if (ec == boost::system::errc::argument_list_too_long) { nlohmann::json metricProperties = nlohmann::json::array(); for (const auto& [uri, _] : uriToDbus) { metricProperties.emplace_back(uri); } messages::propertyValueIncorrect( asyncResp->res, metricProperties, "MetricProperties"); return; } if (ec) { messages::internalError(asyncResp->res); BMCWEB_LOG_ERROR << "respHandler DBus error " << ec; return; } messages::created(asyncResp->res); }, telemetry::service, "/xyz/openbmc_project/Telemetry/Reports", "xyz.openbmc_project.Telemetry.ReportManager", "AddReport", "TelemetryService/" + args.id, args.name, args.reportingType, args.reportUpdates, args.appendLimit, args.reportActions, args.interval, readingParams, args.metricReportDefinitionEnabled); } AddReport(const AddReport&) = delete; AddReport(AddReport&&) = delete; AddReport& operator=(const AddReport&) = delete; AddReport& operator=(AddReport&&) = delete; void insert(const std::map& el) { uriToDbus.insert(el.begin(), el.end()); } private: std::shared_ptr asyncResp; AddReportArgs args; boost::container::flat_map uriToDbus{}; }; } // namespace telemetry inline void handleMetricReportDefinitionCollectionHead( App& app, const crow::Request& req, const std::shared_ptr& asyncResp) { if (!redfish::setUpRedfishRoute(app, req, asyncResp)) { return; } asyncResp->res.addHeader( boost::beast::http::field::link, "; rel=describedby"); } inline void handleMetricReportDefinitionCollectionGet( App& app, const crow::Request& req, const std::shared_ptr& asyncResp) { if (!redfish::setUpRedfishRoute(app, req, asyncResp)) { return; } asyncResp->res.jsonValue["@odata.type"] = "#MetricReportDefinitionCollection." "MetricReportDefinitionCollection"; asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/TelemetryService/MetricReportDefinitions"; asyncResp->res.jsonValue["Name"] = "Metric Definition Collection"; constexpr std::array interfaces{ telemetry::reportInterface}; collection_util::getCollectionMembers( asyncResp, boost::urls::url( "/redfish/v1/TelemetryService/MetricReportDefinitions"), interfaces, "/xyz/openbmc_project/Telemetry/Reports/TelemetryService"); } inline void handleMetricReportHead(App& app, const crow::Request& req, const std::shared_ptr& asyncResp, const std::string& /*id*/) { if (!redfish::setUpRedfishRoute(app, req, asyncResp)) { return; } asyncResp->res.addHeader( boost::beast::http::field::link, "; rel=describedby"); } inline void handleMetricReportGet(App& app, const crow::Request& req, const std::shared_ptr& asyncResp, const std::string& id) { if (!redfish::setUpRedfishRoute(app, req, asyncResp)) { return; } asyncResp->res.addHeader( boost::beast::http::field::link, "; rel=describedby"); sdbusplus::asio::getAllProperties( *crow::connections::systemBus, telemetry::service, telemetry::getDbusReportPath(id), telemetry::reportInterface, [asyncResp, id](const boost::system::error_code& ec, const dbus::utility::DBusPropertiesMap& properties) { if (ec.value() == EBADR || ec == boost::system::errc::host_unreachable) { messages::resourceNotFound(asyncResp->res, "MetricReportDefinition", id); return; } if (ec) { BMCWEB_LOG_ERROR << "respHandler DBus error " << ec; messages::internalError(asyncResp->res); return; } telemetry::fillReportDefinition(asyncResp, id, properties); }); } inline void handleMetricReportDelete( App& 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](const boost::system::error_code& ec) { /* * boost::system::errc and std::errc are missing value * for EBADR error that is defined in Linux. */ if (ec.value() == EBADR) { messages::resourceNotFound(asyncResp->res, "MetricReportDefinition", id); return; } if (ec) { BMCWEB_LOG_ERROR << "respHandler DBus error " << ec; messages::internalError(asyncResp->res); return; } asyncResp->res.result(boost::beast::http::status::no_content); }, telemetry::service, reportPath, "xyz.openbmc_project.Object.Delete", "Delete"); } inline void requestRoutesMetricReportDefinitionCollection(App& app) { BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions/") .privileges(redfish::privileges::headMetricReportDefinitionCollection) .methods(boost::beast::http::verb::head)(std::bind_front( handleMetricReportDefinitionCollectionHead, std::ref(app))); BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions/") .privileges(redfish::privileges::getMetricReportDefinitionCollection) .methods(boost::beast::http::verb::get)(std::bind_front( handleMetricReportDefinitionCollectionGet, std::ref(app))); BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions/") .privileges(redfish::privileges::postMetricReportDefinitionCollection) .methods(boost::beast::http::verb::post)( [&app](const crow::Request& req, const std::shared_ptr& asyncResp) { if (!redfish::setUpRedfishRoute(app, req, asyncResp)) { return; } telemetry::AddReportArgs args; if (!telemetry::getUserParameters(asyncResp->res, req, args)) { return; } boost::container::flat_set> chassisSensors; if (!telemetry::getChassisSensorNodeFromMetrics(asyncResp, args.metrics, chassisSensors)) { return; } auto addReportReq = std::make_shared(std::move(args), asyncResp); for (const auto& [chassis, sensorType] : chassisSensors) { retrieveUriToDbusMap( chassis, sensorType, [asyncResp, addReportReq]( const boost::beast::http::status status, const std::map& uriToDbus) { if (status != boost::beast::http::status::ok) { BMCWEB_LOG_ERROR << "Failed to retrieve URI to dbus sensors map with err " << static_cast(status); return; } addReportReq->insert(uriToDbus); }); } }); } inline void requestRoutesMetricReportDefinition(App& app) { BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions//") .privileges(redfish::privileges::getMetricReportDefinition) .methods(boost::beast::http::verb::head)( std::bind_front(handleMetricReportHead, std::ref(app))); BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions//") .privileges(redfish::privileges::getMetricReportDefinition) .methods(boost::beast::http::verb::get)( std::bind_front(handleMetricReportGet, std::ref(app))); BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions//") .privileges(redfish::privileges::deleteMetricReportDefinitionCollection) .methods(boost::beast::http::verb::delete_)( std::bind_front(handleMetricReportDelete, std::ref(app))); } } // namespace redfish