// SPDX-License-Identifier: Apache-2.0 // SPDX-FileCopyrightText: Copyright OpenBMC Authors #pragma once #include "app.hpp" #include "dbus_utility.hpp" #include "generated/enums/metric_report_definition.hpp" #include "generated/enums/resource.hpp" #include "query.hpp" #include "registries/privilege_registry.hpp" #include "sensors.hpp" #include "utils/collection.hpp" #include "utils/dbus_utils.hpp" #include "utils/json_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 bool verifyCommonErrors(crow::Response& res, const std::string& id, const boost::system::error_code& ec) { if (ec.value() == EBADR || ec == boost::system::errc::host_unreachable) { messages::resourceNotFound(res, "MetricReportDefinition", id); return false; } if (ec == boost::system::errc::file_exists) { messages::resourceAlreadyExists(res, "MetricReportDefinition", "Id", id); return false; } if (ec == boost::system::errc::too_many_files_open) { messages::createLimitReachedForResource(res); return false; } if (ec) { BMCWEB_LOG_ERROR("DBUS response error {}", ec); messages::internalError(res); return false; } return true; } 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"] = resource::State::Enabled; } else { asyncResp->res.jsonValue["Status"]["State"] = resource::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, std::vector& outReportActions) { size_t index = 0; for (const std::string& action : actions) { std::string dbusReportAction = toDbusReportAction(action); if (dbusReportAction.empty()) { messages::propertyValueNotInList( res, action, "ReportActions/" + std::to_string(index)); return false; } outReportActions.emplace_back(std::move(dbusReportAction)); index++; } return true; } inline bool getUserMetric(crow::Response& res, nlohmann::json::object_t& metric, AddReportArgs::MetricArgs& metricArgs) { std::optional> uris; std::optional collectionDurationStr; std::optional collectionFunction; std::optional collectionTimeScopeStr; if (!json_util::readJsonObject( // metric, res, // "CollectionDuration", collectionDurationStr, // "CollectionFunction", collectionFunction, // "CollectionTimeScope", collectionTimeScopeStr, // "MetricProperties", uris // )) { 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::object_t& 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 scheduleDurationStr; if (!json_util::readJsonPatch( // req, res, // "AppendLimit", appendLimit, // "Id", id, // "MetricReportDefinitionEnabled", metricReportDefinitionEnabled, // "MetricReportDefinitionType", reportingTypeStr, // "Metrics", metrics, // "Name", name, // "ReportActions", reportActionsStr, // "ReportUpdates", reportUpdatesStr, // "Schedule/RecurrenceInterval", scheduleDurationStr // )) { 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.reportActions)) { return false; } } if (reportingTypeStr == "Periodic") { if (!scheduleDurationStr) { messages::createFailedMissingReqProperties(res, "Schedule"); return false; } std::optional durationNum = time_utils::fromDurationString(*scheduleDurationStr); if (!durationNum || durationNum->count() < 0) { messages::propertyValueIncorrect(res, "RecurrenceInterval", *scheduleDurationStr); 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; } inline std::string toRedfishProperty(std::string_view dbusMessage) { if (dbusMessage == "Id") { return "Id"; } if (dbusMessage == "Name") { return "Name"; } if (dbusMessage == "ReportingType") { return "MetricReportDefinitionType"; } if (dbusMessage == "AppendLimit") { return "AppendLimit"; } if (dbusMessage == "ReportActions") { return "ReportActions"; } if (dbusMessage == "Interval") { return "RecurrenceInterval"; } if (dbusMessage == "ReportUpdates") { return "ReportUpdates"; } if (dbusMessage == "ReadingParameters") { return "Metrics"; } return ""; } inline bool handleParamError(crow::Response& res, const char* errorMessage, std::string_view key) { if (errorMessage == nullptr) { BMCWEB_LOG_ERROR("errorMessage was null"); return true; } std::string_view errorMessageSv(errorMessage); if (errorMessageSv.starts_with(key)) { std::string redfishProperty = toRedfishProperty(key); if (redfishProperty.empty()) { // Getting here means most possibly that toRedfishProperty has // incomplete implementation. Return internal error for now. BMCWEB_LOG_ERROR("{} has no corresponding Redfish property", key); messages::internalError(res); return false; } messages::propertyValueError(res, redfishProperty); return false; } return true; } inline void afterAddReport(const std::shared_ptr& asyncResp, const AddReportArgs& args, const boost::system::error_code& ec, const sdbusplus::message_t& msg) { if (!ec) { messages::created(asyncResp->res); return; } if (ec == boost::system::errc::invalid_argument) { const sd_bus_error* errorMessage = msg.get_error(); if (errorMessage != nullptr) { for (const auto& arg : {"Id", "Name", "ReportingType", "AppendLimit", "ReportActions", "Interval", "ReportUpdates", "ReadingParameters"}) { if (!handleParamError(asyncResp->res, errorMessage->message, arg)) { return; } } } } if (!verifyCommonErrors(asyncResp->res, args.id, ec)) { return; } messages::internalError(asyncResp->res); } 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, args](const boost::system::error_code& ec, const sdbusplus::message_t& msg, const std::string& /*arg1*/) { afterAddReport(asyncResp, args, ec, msg); }, 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: const std::shared_ptr asyncResp; AddReportArgs args; boost::container::flat_map uriToDbus; }; inline std::optional< std::vector>> sensorPathToUri( const std::shared_ptr& asyncResp, std::span uris, const std::map& metricPropertyToDbusPaths) { std::vector> result; for (const std::string& uri : uris) { auto it = metricPropertyToDbusPaths.find(uri); if (it == metricPropertyToDbusPaths.end()) { messages::propertyValueNotInList(asyncResp->res, uri, "MetricProperties"); return {}; } result.emplace_back(it->second, uri); } return result; } inline void afterSetReadingParams( const std::shared_ptr& asyncResp, const std::string& reportId, const boost::system::error_code& ec, const sdbusplus::message_t& msg) { if (!ec) { messages::success(asyncResp->res); return; } if (ec == boost::system::errc::invalid_argument) { const sd_bus_error* errorMessage = msg.get_error(); if (errorMessage != nullptr) { for (const auto& arg : {"Id", "ReadingParameters"}) { if (!handleParamError(asyncResp->res, errorMessage->message, arg)) { return; } } } } if (!verifyCommonErrors(asyncResp->res, reportId, ec)) { return; } messages::internalError(asyncResp->res); } inline void setReadingParams( const std::shared_ptr& asyncResp, const std::string& reportId, ReadingParameters readingParams, const std::vector>& readingParamsUris, const std::map& metricPropertyToDbusPaths) { if (asyncResp->res.result() != boost::beast::http::status::ok) { return; } for (size_t index = 0; index < readingParamsUris.size(); ++index) { std::span newUris = readingParamsUris[index]; const std::optional>> readingParam = sensorPathToUri(asyncResp, newUris, metricPropertyToDbusPaths); if (!readingParam) { return; } for (const std::tuple& value : *readingParam) { std::get<0>(readingParams[index]).emplace_back(value); } } crow::connections::systemBus->async_method_call( [asyncResp, reportId](const boost::system::error_code& ec, const sdbusplus::message_t& msg) { afterSetReadingParams(asyncResp, reportId, ec, msg); }, "xyz.openbmc_project.Telemetry", getDbusReportPath(reportId), "org.freedesktop.DBus.Properties", "Set", "xyz.openbmc_project.Telemetry.Report", "ReadingParameters", dbus::utility::DbusVariantType{readingParams}); } class UpdateMetrics { public: UpdateMetrics(std::string_view idIn, const std::shared_ptr& asyncRespIn) : id(idIn), asyncResp(asyncRespIn) {} ~UpdateMetrics() { boost::asio::post( crow::connections::systemBus->get_io_context(), std::bind_front(&setReadingParams, asyncResp, id, std::move(readingParams), readingParamsUris, metricPropertyToDbusPaths)); } UpdateMetrics(const UpdateMetrics&) = delete; UpdateMetrics(UpdateMetrics&&) = delete; UpdateMetrics& operator=(const UpdateMetrics&) = delete; UpdateMetrics& operator=(UpdateMetrics&&) = delete; std::string id; std::map metricPropertyToDbusPaths; void insert(const std::map& additionalMetricPropertyToDbusPaths) { metricPropertyToDbusPaths.insert( additionalMetricPropertyToDbusPaths.begin(), additionalMetricPropertyToDbusPaths.end()); } void emplace( std::span< const std::tuple> pathAndUri, const AddReportArgs::MetricArgs& metricArgs) { readingParamsUris.emplace_back(metricArgs.uris); readingParams.emplace_back( std::vector(pathAndUri.begin(), pathAndUri.end()), metricArgs.collectionFunction, metricArgs.collectionTimeScope, metricArgs.collectionDuration); } private: const std::shared_ptr asyncResp; std::vector> readingParamsUris; ReadingParameters readingParams; }; inline void setReportEnabled(const std::shared_ptr& asyncResp, std::string_view id, bool enabled) { crow::connections::systemBus->async_method_call( [asyncResp, id = std::string(id)](const boost::system::error_code& ec) { if (!verifyCommonErrors(asyncResp->res, id, ec)) { return; } }, "xyz.openbmc_project.Telemetry", getDbusReportPath(id), "org.freedesktop.DBus.Properties", "Set", "xyz.openbmc_project.Telemetry.Report", "Enabled", dbus::utility::DbusVariantType{enabled}); } inline void afterSetReportingProperties( const std::shared_ptr& asyncResp, const std::string& id, const boost::system::error_code& ec, const sdbusplus::message_t& msg) { if (!ec) { asyncResp->res.result(boost::beast::http::status::no_content); return; } if (ec == boost::system::errc::invalid_argument) { const sd_bus_error* errorMessage = msg.get_error(); if (errorMessage != nullptr) { for (const auto& arg : {"Id", "ReportingType", "Interval"}) { if (!handleParamError(asyncResp->res, errorMessage->message, arg)) { return; } } } } if (!verifyCommonErrors(asyncResp->res, id, ec)) { return; } messages::internalError(asyncResp->res); } inline void setReportTypeAndInterval( const std::shared_ptr& asyncResp, std::string_view id, const std::optional& reportingType, const std::optional& recurrenceIntervalStr) { std::string dbusReportingType; if (reportingType) { dbusReportingType = toDbusReportingType(*reportingType); if (dbusReportingType.empty()) { messages::propertyValueNotInList(asyncResp->res, *reportingType, "MetricReportDefinitionType"); return; } } uint64_t recurrenceInterval = std::numeric_limits::max(); if (recurrenceIntervalStr) { std::optional durationNum = time_utils::fromDurationString(*recurrenceIntervalStr); if (!durationNum || durationNum->count() < 0) { messages::propertyValueIncorrect( asyncResp->res, "RecurrenceInterval", *recurrenceIntervalStr); return; } recurrenceInterval = static_cast(durationNum->count()); } crow::connections::systemBus->async_method_call( [asyncResp, id = std::string(id)](const boost::system::error_code& ec, const sdbusplus::message_t& msg) { afterSetReportingProperties(asyncResp, id, ec, msg); }, "xyz.openbmc_project.Telemetry", getDbusReportPath(id), "xyz.openbmc_project.Telemetry.Report", "SetReportingProperties", dbusReportingType, recurrenceInterval); } inline void afterSetReportUpdates( const std::shared_ptr& asyncResp, const std::string& id, const boost::system::error_code& ec, const sdbusplus::message_t& msg) { if (!ec) { asyncResp->res.result(boost::beast::http::status::no_content); return; } if (ec == boost::system::errc::invalid_argument) { const sd_bus_error* errorMessage = msg.get_error(); if (errorMessage != nullptr) { for (const auto& arg : {"Id", "ReportUpdates"}) { if (!handleParamError(asyncResp->res, errorMessage->message, arg)) { return; } } } } if (!verifyCommonErrors(asyncResp->res, id, ec)) { return; } } inline void setReportUpdates(const std::shared_ptr& asyncResp, std::string_view id, const std::string& reportUpdates) { std::string dbusReportUpdates = toDbusReportUpdates(reportUpdates); if (dbusReportUpdates.empty()) { messages::propertyValueNotInList(asyncResp->res, reportUpdates, "ReportUpdates"); return; } crow::connections::systemBus->async_method_call( [asyncResp, id = std::string(id)](const boost::system::error_code& ec, const sdbusplus::message_t& msg) { afterSetReportUpdates(asyncResp, id, ec, msg); }, "xyz.openbmc_project.Telemetry", getDbusReportPath(id), "org.freedesktop.DBus.Properties", "Set", "xyz.openbmc_project.Telemetry.Report", "ReportUpdates", dbus::utility::DbusVariantType{dbusReportUpdates}); } inline void afterSetReportActions( const std::shared_ptr& asyncResp, const std::string& id, const boost::system::error_code& ec, const sdbusplus::message_t& msg) { if (ec == boost::system::errc::invalid_argument) { const sd_bus_error* errorMessage = msg.get_error(); if (errorMessage != nullptr) { for (const auto& arg : {"Id", "ReportActions"}) { if (!handleParamError(asyncResp->res, errorMessage->message, arg)) { return; } } } } if (!verifyCommonErrors(asyncResp->res, id, ec)) { return; } messages::internalError(asyncResp->res); } inline void setReportActions( const std::shared_ptr& asyncResp, std::string_view id, const std::vector& reportActions) { std::vector dbusReportActions; if (!toDbusReportActions(asyncResp->res, reportActions, dbusReportActions)) { return; } crow::connections::systemBus->async_method_call( [asyncResp, id = std::string(id)](const boost::system::error_code& ec, const sdbusplus::message_t& msg) { afterSetReportActions(asyncResp, id, ec, msg); }, "xyz.openbmc_project.Telemetry", getDbusReportPath(id), "org.freedesktop.DBus.Properties", "Set", "xyz.openbmc_project.Telemetry.Report", "ReportActions", dbus::utility::DbusVariantType{dbusReportActions}); } inline void setReportMetrics( const std::shared_ptr& asyncResp, std::string_view id, std::vector>&& metrics) { dbus::utility::getAllProperties( telemetry::service, telemetry::getDbusReportPath(id), telemetry::reportInterface, [asyncResp, id = std::string(id), redfishMetrics = std::move(metrics)]( boost::system::error_code ec, const dbus::utility::DBusPropertiesMap& properties) mutable { if (!verifyCommonErrors(asyncResp->res, id, ec)) { return; } ReadingParameters readingParams; const bool success = sdbusplus::unpackPropertiesNoThrow( dbus_utils::UnpackErrorPrinter(), properties, "ReadingParameters", readingParams); if (!success) { messages::internalError(asyncResp->res); return; } auto updateMetricsReq = std::make_shared(id, asyncResp); boost::container::flat_set> chassisSensors; size_t index = 0; for (std::variant& metricVariant : redfishMetrics) { nlohmann::json::object_t* metric = std::get_if(&metricVariant); if (metric == nullptr) { index++; continue; } AddReportArgs::MetricArgs metricArgs; std::vector< std::tuple> pathAndUri; if (index < readingParams.size()) { const ReadingParameters::value_type& existing = readingParams[index]; if (metric->empty()) { pathAndUri = std::get<0>(existing); } metricArgs.collectionFunction = std::get<1>(existing); metricArgs.collectionTimeScope = std::get<2>(existing); metricArgs.collectionDuration = std::get<3>(existing); } if (!getUserMetric(asyncResp->res, *metric, metricArgs)) { return; } std::optional error = getChassisSensorNode(metricArgs.uris, chassisSensors); if (error) { messages::propertyValueIncorrect( asyncResp->res, error->uri, "MetricProperties/" + std::to_string(error->index)); return; } updateMetricsReq->emplace(pathAndUri, metricArgs); index++; } for (const auto& [chassis, sensorType] : chassisSensors) { retrieveUriToDbusMap( chassis, sensorType, [asyncResp, updateMetricsReq]( 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; } updateMetricsReq->insert(uriToDbus); }); } }); } 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.addHeader( boost::beast::http::field::link, "; rel=describedby"); 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 handleReportPatch( App& app, const crow::Request& req, const std::shared_ptr& asyncResp, std::string_view id) { if (!redfish::setUpRedfishRoute(app, req, asyncResp)) { return; } std::optional reportingTypeStr; std::optional reportUpdatesStr; std::optional metricReportDefinitionEnabled; std::optional< std::vector>> metrics; std::optional> reportActionsStr; std::optional scheduleDurationStr; if (!json_util::readJsonPatch( // req, asyncResp->res, // "MetricReportDefinitionEnabled", metricReportDefinitionEnabled, // "MetricReportDefinitionType", reportingTypeStr, // "Metrics", metrics, // "ReportActions", reportActionsStr, // "ReportUpdates", reportUpdatesStr, // "Schedule/RecurrenceInterval", scheduleDurationStr // )) { return; } if (metricReportDefinitionEnabled) { setReportEnabled(asyncResp, id, *metricReportDefinitionEnabled); } if (reportUpdatesStr) { setReportUpdates(asyncResp, id, *reportUpdatesStr); } if (reportActionsStr) { setReportActions(asyncResp, id, *reportActionsStr); } if (reportingTypeStr || scheduleDurationStr) { setReportTypeAndInterval(asyncResp, id, reportingTypeStr, scheduleDurationStr); } if (metrics) { setReportMetrics(asyncResp, id, std::move(*metrics)); } } inline void handleReportDelete( App& app, const crow::Request& req, const std::shared_ptr& asyncResp, std::string_view id) { if (!redfish::setUpRedfishRoute(app, req, asyncResp)) { return; } const std::string reportPath = getDbusReportPath(id); crow::connections::systemBus->async_method_call( [asyncResp, reportId = std::string(id)](const boost::system::error_code& ec) { if (!verifyCommonErrors(asyncResp->res, reportId, ec)) { return; } asyncResp->res.result(boost::beast::http::status::no_content); }, service, reportPath, "xyz.openbmc_project.Object.Delete", "Delete"); } } // namespace telemetry inline void afterRetrieveUriToDbusMap( const std::shared_ptr& /*asyncResp*/, const std::shared_ptr& 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 handleMetricReportDefinitionsPost( App& 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, std::bind_front(afterRetrieveUriToDbusMap, asyncResp, addReportReq)); } } 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"); dbus::utility::getAllProperties( telemetry::service, telemetry::getDbusReportPath(id), telemetry::reportInterface, [asyncResp, id](const boost::system::error_code& ec, const dbus::utility::DBusPropertiesMap& properties) { if (!redfish::telemetry::verifyCommonErrors(asyncResp->res, id, ec)) { 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( telemetry::handleMetricReportDefinitionCollectionHead, std::ref(app))); BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions/") .privileges(redfish::privileges::getMetricReportDefinitionCollection) .methods(boost::beast::http::verb::get)(std::bind_front( telemetry::handleMetricReportDefinitionCollectionGet, std::ref(app))); BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions/") .privileges(redfish::privileges::postMetricReportDefinitionCollection) .methods(boost::beast::http::verb::post)( std::bind_front(handleMetricReportDefinitionsPost, std::ref(app))); } 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::deleteMetricReportDefinition) .methods(boost::beast::http::verb::delete_)( std::bind_front(handleMetricReportDelete, std::ref(app))); BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions//") .privileges(redfish::privileges::patchMetricReportDefinition) .methods(boost::beast::http::verb::patch)( std::bind_front(telemetry::handleReportPatch, std::ref(app))); } } // namespace redfish