1081ebf06SWludzik, Jozef #pragma once 2081ebf06SWludzik, Jozef 33ccb3adbSEd Tanous #include "app.hpp" 43ccb3adbSEd Tanous #include "dbus_utility.hpp" 5479e899dSKrzysztof Grobelny #include "generated/enums/metric_report_definition.hpp" 63ccb3adbSEd Tanous #include "query.hpp" 73ccb3adbSEd Tanous #include "registries/privilege_registry.hpp" 84dbb8aeaSWludzik, Jozef #include "sensors.hpp" 93ccb3adbSEd Tanous #include "utils/collection.hpp" 103ccb3adbSEd Tanous #include "utils/dbus_utils.hpp" 11081ebf06SWludzik, Jozef #include "utils/telemetry_utils.hpp" 12081ebf06SWludzik, Jozef #include "utils/time_utils.hpp" 13081ebf06SWludzik, Jozef 144dbb8aeaSWludzik, Jozef #include <boost/container/flat_map.hpp> 15ef4c65b7SEd Tanous #include <boost/url/format.hpp> 1689474494SKrzysztof Grobelny #include <sdbusplus/asio/property.hpp> 1789474494SKrzysztof Grobelny #include <sdbusplus/unpack_properties.hpp> 184dbb8aeaSWludzik, Jozef 197a1dbc48SGeorge Liu #include <array> 20fe04d49cSNan Zhou #include <map> 21f19ab44aSSzymon Dompke #include <optional> 22f19ab44aSSzymon Dompke #include <span> 23f19ab44aSSzymon Dompke #include <string> 247a1dbc48SGeorge Liu #include <string_view> 25081ebf06SWludzik, Jozef #include <tuple> 26f19ab44aSSzymon Dompke #include <utility> 27081ebf06SWludzik, Jozef #include <variant> 28f19ab44aSSzymon Dompke #include <vector> 29081ebf06SWludzik, Jozef 30081ebf06SWludzik, Jozef namespace redfish 31081ebf06SWludzik, Jozef { 32081ebf06SWludzik, Jozef 33081ebf06SWludzik, Jozef namespace telemetry 34081ebf06SWludzik, Jozef { 35081ebf06SWludzik, Jozef 36479e899dSKrzysztof Grobelny using ReadingParameters = std::vector<std::tuple< 37479e899dSKrzysztof Grobelny std::vector<std::tuple<sdbusplus::message::object_path, std::string>>, 38479e899dSKrzysztof Grobelny std::string, std::string, uint64_t>>; 39479e899dSKrzysztof Grobelny 409e6c388aSLukasz Kazmierczak inline bool verifyCommonErrors(crow::Response& res, const std::string& id, 419e6c388aSLukasz Kazmierczak const boost::system::error_code& ec) 429e6c388aSLukasz Kazmierczak { 439e6c388aSLukasz Kazmierczak if (ec.value() == EBADR || ec == boost::system::errc::host_unreachable) 449e6c388aSLukasz Kazmierczak { 459e6c388aSLukasz Kazmierczak messages::resourceNotFound(res, "MetricReportDefinition", id); 469e6c388aSLukasz Kazmierczak return false; 479e6c388aSLukasz Kazmierczak } 489e6c388aSLukasz Kazmierczak 499e6c388aSLukasz Kazmierczak if (ec == boost::system::errc::file_exists) 509e6c388aSLukasz Kazmierczak { 519e6c388aSLukasz Kazmierczak messages::resourceAlreadyExists(res, "MetricReportDefinition", "Id", 529e6c388aSLukasz Kazmierczak id); 539e6c388aSLukasz Kazmierczak return false; 549e6c388aSLukasz Kazmierczak } 559e6c388aSLukasz Kazmierczak 569e6c388aSLukasz Kazmierczak if (ec == boost::system::errc::too_many_files_open) 579e6c388aSLukasz Kazmierczak { 589e6c388aSLukasz Kazmierczak messages::createLimitReachedForResource(res); 599e6c388aSLukasz Kazmierczak return false; 609e6c388aSLukasz Kazmierczak } 619e6c388aSLukasz Kazmierczak 629e6c388aSLukasz Kazmierczak if (ec) 639e6c388aSLukasz Kazmierczak { 649e6c388aSLukasz Kazmierczak BMCWEB_LOG_ERROR("DBUS response error {}", ec); 659e6c388aSLukasz Kazmierczak messages::internalError(res); 669e6c388aSLukasz Kazmierczak return false; 679e6c388aSLukasz Kazmierczak } 689e6c388aSLukasz Kazmierczak 699e6c388aSLukasz Kazmierczak return true; 709e6c388aSLukasz Kazmierczak } 719e6c388aSLukasz Kazmierczak 72479e899dSKrzysztof Grobelny inline metric_report_definition::ReportActionsEnum 73479e899dSKrzysztof Grobelny toRedfishReportAction(std::string_view dbusValue) 74479e899dSKrzysztof Grobelny { 75479e899dSKrzysztof Grobelny if (dbusValue == 76479e899dSKrzysztof Grobelny "xyz.openbmc_project.Telemetry.Report.ReportActions.EmitsReadingsUpdate") 77479e899dSKrzysztof Grobelny { 78479e899dSKrzysztof Grobelny return metric_report_definition::ReportActionsEnum::RedfishEvent; 79479e899dSKrzysztof Grobelny } 80479e899dSKrzysztof Grobelny if (dbusValue == 81479e899dSKrzysztof Grobelny "xyz.openbmc_project.Telemetry.Report.ReportActions.LogToMetricReportsCollection") 82479e899dSKrzysztof Grobelny { 83479e899dSKrzysztof Grobelny return metric_report_definition::ReportActionsEnum:: 84479e899dSKrzysztof Grobelny LogToMetricReportsCollection; 85479e899dSKrzysztof Grobelny } 86479e899dSKrzysztof Grobelny return metric_report_definition::ReportActionsEnum::Invalid; 87479e899dSKrzysztof Grobelny } 88479e899dSKrzysztof Grobelny 89479e899dSKrzysztof Grobelny inline std::string toDbusReportAction(std::string_view redfishValue) 90479e899dSKrzysztof Grobelny { 91479e899dSKrzysztof Grobelny if (redfishValue == "RedfishEvent") 92479e899dSKrzysztof Grobelny { 93479e899dSKrzysztof Grobelny return "xyz.openbmc_project.Telemetry.Report.ReportActions.EmitsReadingsUpdate"; 94479e899dSKrzysztof Grobelny } 95479e899dSKrzysztof Grobelny if (redfishValue == "LogToMetricReportsCollection") 96479e899dSKrzysztof Grobelny { 97479e899dSKrzysztof Grobelny return "xyz.openbmc_project.Telemetry.Report.ReportActions.LogToMetricReportsCollection"; 98479e899dSKrzysztof Grobelny } 99479e899dSKrzysztof Grobelny return ""; 100479e899dSKrzysztof Grobelny } 101479e899dSKrzysztof Grobelny 102479e899dSKrzysztof Grobelny inline metric_report_definition::MetricReportDefinitionType 103479e899dSKrzysztof Grobelny toRedfishReportingType(std::string_view dbusValue) 104479e899dSKrzysztof Grobelny { 105479e899dSKrzysztof Grobelny if (dbusValue == 106479e899dSKrzysztof Grobelny "xyz.openbmc_project.Telemetry.Report.ReportingType.OnChange") 107479e899dSKrzysztof Grobelny { 108479e899dSKrzysztof Grobelny return metric_report_definition::MetricReportDefinitionType::OnChange; 109479e899dSKrzysztof Grobelny } 110479e899dSKrzysztof Grobelny if (dbusValue == 111479e899dSKrzysztof Grobelny "xyz.openbmc_project.Telemetry.Report.ReportingType.OnRequest") 112479e899dSKrzysztof Grobelny { 113479e899dSKrzysztof Grobelny return metric_report_definition::MetricReportDefinitionType::OnRequest; 114479e899dSKrzysztof Grobelny } 115479e899dSKrzysztof Grobelny if (dbusValue == 116479e899dSKrzysztof Grobelny "xyz.openbmc_project.Telemetry.Report.ReportingType.Periodic") 117479e899dSKrzysztof Grobelny { 118479e899dSKrzysztof Grobelny return metric_report_definition::MetricReportDefinitionType::Periodic; 119479e899dSKrzysztof Grobelny } 120479e899dSKrzysztof Grobelny return metric_report_definition::MetricReportDefinitionType::Invalid; 121479e899dSKrzysztof Grobelny } 122479e899dSKrzysztof Grobelny 123479e899dSKrzysztof Grobelny inline std::string toDbusReportingType(std::string_view redfishValue) 124479e899dSKrzysztof Grobelny { 125479e899dSKrzysztof Grobelny if (redfishValue == "OnChange") 126479e899dSKrzysztof Grobelny { 127479e899dSKrzysztof Grobelny return "xyz.openbmc_project.Telemetry.Report.ReportingType.OnChange"; 128479e899dSKrzysztof Grobelny } 129479e899dSKrzysztof Grobelny if (redfishValue == "OnRequest") 130479e899dSKrzysztof Grobelny { 131479e899dSKrzysztof Grobelny return "xyz.openbmc_project.Telemetry.Report.ReportingType.OnRequest"; 132479e899dSKrzysztof Grobelny } 133479e899dSKrzysztof Grobelny if (redfishValue == "Periodic") 134479e899dSKrzysztof Grobelny { 135479e899dSKrzysztof Grobelny return "xyz.openbmc_project.Telemetry.Report.ReportingType.Periodic"; 136479e899dSKrzysztof Grobelny } 137479e899dSKrzysztof Grobelny return ""; 138479e899dSKrzysztof Grobelny } 139479e899dSKrzysztof Grobelny 140479e899dSKrzysztof Grobelny inline metric_report_definition::CollectionTimeScope 141479e899dSKrzysztof Grobelny toRedfishCollectionTimeScope(std::string_view dbusValue) 142479e899dSKrzysztof Grobelny { 143479e899dSKrzysztof Grobelny if (dbusValue == 144479e899dSKrzysztof Grobelny "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.Point") 145479e899dSKrzysztof Grobelny { 146479e899dSKrzysztof Grobelny return metric_report_definition::CollectionTimeScope::Point; 147479e899dSKrzysztof Grobelny } 148479e899dSKrzysztof Grobelny if (dbusValue == 149479e899dSKrzysztof Grobelny "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.Interval") 150479e899dSKrzysztof Grobelny { 151479e899dSKrzysztof Grobelny return metric_report_definition::CollectionTimeScope::Interval; 152479e899dSKrzysztof Grobelny } 153479e899dSKrzysztof Grobelny if (dbusValue == 154479e899dSKrzysztof Grobelny "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.StartupInterval") 155479e899dSKrzysztof Grobelny { 156479e899dSKrzysztof Grobelny return metric_report_definition::CollectionTimeScope::StartupInterval; 157479e899dSKrzysztof Grobelny } 158479e899dSKrzysztof Grobelny return metric_report_definition::CollectionTimeScope::Invalid; 159479e899dSKrzysztof Grobelny } 160479e899dSKrzysztof Grobelny 161479e899dSKrzysztof Grobelny inline std::string toDbusCollectionTimeScope(std::string_view redfishValue) 162479e899dSKrzysztof Grobelny { 163479e899dSKrzysztof Grobelny if (redfishValue == "Point") 164479e899dSKrzysztof Grobelny { 165479e899dSKrzysztof Grobelny return "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.Point"; 166479e899dSKrzysztof Grobelny } 167479e899dSKrzysztof Grobelny if (redfishValue == "Interval") 168479e899dSKrzysztof Grobelny { 169479e899dSKrzysztof Grobelny return "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.Interval"; 170479e899dSKrzysztof Grobelny } 171479e899dSKrzysztof Grobelny if (redfishValue == "StartupInterval") 172479e899dSKrzysztof Grobelny { 173479e899dSKrzysztof Grobelny return "xyz.openbmc_project.Telemetry.Report.CollectionTimescope.StartupInterval"; 174479e899dSKrzysztof Grobelny } 175479e899dSKrzysztof Grobelny return ""; 176479e899dSKrzysztof Grobelny } 177479e899dSKrzysztof Grobelny 178479e899dSKrzysztof Grobelny inline metric_report_definition::ReportUpdatesEnum 179479e899dSKrzysztof Grobelny toRedfishReportUpdates(std::string_view dbusValue) 180479e899dSKrzysztof Grobelny { 181479e899dSKrzysztof Grobelny if (dbusValue == 182479e899dSKrzysztof Grobelny "xyz.openbmc_project.Telemetry.Report.ReportUpdates.Overwrite") 183479e899dSKrzysztof Grobelny { 184479e899dSKrzysztof Grobelny return metric_report_definition::ReportUpdatesEnum::Overwrite; 185479e899dSKrzysztof Grobelny } 186479e899dSKrzysztof Grobelny if (dbusValue == 187479e899dSKrzysztof Grobelny "xyz.openbmc_project.Telemetry.Report.ReportUpdates.AppendWrapsWhenFull") 188479e899dSKrzysztof Grobelny { 189479e899dSKrzysztof Grobelny return metric_report_definition::ReportUpdatesEnum::AppendWrapsWhenFull; 190479e899dSKrzysztof Grobelny } 191479e899dSKrzysztof Grobelny if (dbusValue == 192479e899dSKrzysztof Grobelny "xyz.openbmc_project.Telemetry.Report.ReportUpdates.AppendStopsWhenFull") 193479e899dSKrzysztof Grobelny { 194479e899dSKrzysztof Grobelny return metric_report_definition::ReportUpdatesEnum::AppendStopsWhenFull; 195479e899dSKrzysztof Grobelny } 196479e899dSKrzysztof Grobelny return metric_report_definition::ReportUpdatesEnum::Invalid; 197479e899dSKrzysztof Grobelny } 198479e899dSKrzysztof Grobelny 199479e899dSKrzysztof Grobelny inline std::string toDbusReportUpdates(std::string_view redfishValue) 200479e899dSKrzysztof Grobelny { 201479e899dSKrzysztof Grobelny if (redfishValue == "Overwrite") 202479e899dSKrzysztof Grobelny { 203479e899dSKrzysztof Grobelny return "xyz.openbmc_project.Telemetry.Report.ReportUpdates.Overwrite"; 204479e899dSKrzysztof Grobelny } 205479e899dSKrzysztof Grobelny if (redfishValue == "AppendWrapsWhenFull") 206479e899dSKrzysztof Grobelny { 207479e899dSKrzysztof Grobelny return "xyz.openbmc_project.Telemetry.Report.ReportUpdates.AppendWrapsWhenFull"; 208479e899dSKrzysztof Grobelny } 209479e899dSKrzysztof Grobelny if (redfishValue == "AppendStopsWhenFull") 210479e899dSKrzysztof Grobelny { 211479e899dSKrzysztof Grobelny return "xyz.openbmc_project.Telemetry.Report.ReportUpdates.AppendStopsWhenFull"; 212479e899dSKrzysztof Grobelny } 213479e899dSKrzysztof Grobelny return ""; 214479e899dSKrzysztof Grobelny } 215081ebf06SWludzik, Jozef 216f19ab44aSSzymon Dompke inline std::optional<nlohmann::json::array_t> getLinkedTriggers( 217f19ab44aSSzymon Dompke std::span<const sdbusplus::message::object_path> triggerPaths) 218f19ab44aSSzymon Dompke { 219f19ab44aSSzymon Dompke nlohmann::json::array_t triggers; 220f19ab44aSSzymon Dompke 221f19ab44aSSzymon Dompke for (const sdbusplus::message::object_path& path : triggerPaths) 222f19ab44aSSzymon Dompke { 223f19ab44aSSzymon Dompke if (path.parent_path() != 224f19ab44aSSzymon Dompke "/xyz/openbmc_project/Telemetry/Triggers/TelemetryService") 225f19ab44aSSzymon Dompke { 22662598e31SEd Tanous BMCWEB_LOG_ERROR("Property Triggers contains invalid value: {}", 22762598e31SEd Tanous path.str); 228f19ab44aSSzymon Dompke return std::nullopt; 229f19ab44aSSzymon Dompke } 230f19ab44aSSzymon Dompke 231f19ab44aSSzymon Dompke std::string id = path.filename(); 232f19ab44aSSzymon Dompke if (id.empty()) 233f19ab44aSSzymon Dompke { 23462598e31SEd Tanous BMCWEB_LOG_ERROR("Property Triggers contains invalid value: {}", 23562598e31SEd Tanous path.str); 236f19ab44aSSzymon Dompke return std::nullopt; 237f19ab44aSSzymon Dompke } 238f19ab44aSSzymon Dompke nlohmann::json::object_t trigger; 239f19ab44aSSzymon Dompke trigger["@odata.id"] = 240f19ab44aSSzymon Dompke boost::urls::format("/redfish/v1/TelemetryService/Triggers/{}", id); 241f19ab44aSSzymon Dompke triggers.emplace_back(std::move(trigger)); 242f19ab44aSSzymon Dompke } 243f19ab44aSSzymon Dompke 244f19ab44aSSzymon Dompke return triggers; 245f19ab44aSSzymon Dompke } 246f19ab44aSSzymon Dompke 247b9d36b47SEd Tanous inline void 248b9d36b47SEd Tanous fillReportDefinition(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 249b9d36b47SEd Tanous const std::string& id, 250479e899dSKrzysztof Grobelny const dbus::utility::DBusPropertiesMap& properties) 251081ebf06SWludzik, Jozef { 252479e899dSKrzysztof Grobelny std::vector<std::string> reportActions; 253479e899dSKrzysztof Grobelny ReadingParameters readingParams; 254479e899dSKrzysztof Grobelny std::string reportingType; 255479e899dSKrzysztof Grobelny std::string reportUpdates; 256479e899dSKrzysztof Grobelny std::string name; 257479e899dSKrzysztof Grobelny uint64_t appendLimit = 0; 258479e899dSKrzysztof Grobelny uint64_t interval = 0; 259479e899dSKrzysztof Grobelny bool enabled = false; 260f19ab44aSSzymon Dompke std::vector<sdbusplus::message::object_path> triggers; 26189474494SKrzysztof Grobelny 26289474494SKrzysztof Grobelny const bool success = sdbusplus::unpackPropertiesNoThrow( 263479e899dSKrzysztof Grobelny dbus_utils::UnpackErrorPrinter(), properties, "ReportingType", 264479e899dSKrzysztof Grobelny reportingType, "Interval", interval, "ReportActions", reportActions, 265479e899dSKrzysztof Grobelny "ReportUpdates", reportUpdates, "AppendLimit", appendLimit, 266f19ab44aSSzymon Dompke "ReadingParameters", readingParams, "Name", name, "Enabled", enabled, 267f19ab44aSSzymon Dompke "Triggers", triggers); 26889474494SKrzysztof Grobelny 26989474494SKrzysztof Grobelny if (!success) 270081ebf06SWludzik, Jozef { 271081ebf06SWludzik, Jozef messages::internalError(asyncResp->res); 272081ebf06SWludzik, Jozef return; 273081ebf06SWludzik, Jozef } 274081ebf06SWludzik, Jozef 275479e899dSKrzysztof Grobelny metric_report_definition::MetricReportDefinitionType redfishReportingType = 276479e899dSKrzysztof Grobelny toRedfishReportingType(reportingType); 277479e899dSKrzysztof Grobelny if (redfishReportingType == 278479e899dSKrzysztof Grobelny metric_report_definition::MetricReportDefinitionType::Invalid) 279081ebf06SWludzik, Jozef { 280479e899dSKrzysztof Grobelny messages::internalError(asyncResp->res); 281479e899dSKrzysztof Grobelny return; 282081ebf06SWludzik, Jozef } 28389474494SKrzysztof Grobelny 284479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["MetricReportDefinitionType"] = 285479e899dSKrzysztof Grobelny redfishReportingType; 286479e899dSKrzysztof Grobelny 287f19ab44aSSzymon Dompke std::optional<nlohmann::json::array_t> linkedTriggers = 288f19ab44aSSzymon Dompke getLinkedTriggers(triggers); 289f19ab44aSSzymon Dompke if (!linkedTriggers) 290f19ab44aSSzymon Dompke { 291f19ab44aSSzymon Dompke messages::internalError(asyncResp->res); 292f19ab44aSSzymon Dompke return; 293f19ab44aSSzymon Dompke } 294f19ab44aSSzymon Dompke 295f19ab44aSSzymon Dompke asyncResp->res.jsonValue["Links"]["Triggers"] = std::move(*linkedTriggers); 296f19ab44aSSzymon Dompke 297479e899dSKrzysztof Grobelny nlohmann::json::array_t redfishReportActions; 298479e899dSKrzysztof Grobelny for (const std::string& action : reportActions) 299081ebf06SWludzik, Jozef { 300479e899dSKrzysztof Grobelny metric_report_definition::ReportActionsEnum redfishAction = 301479e899dSKrzysztof Grobelny toRedfishReportAction(action); 302479e899dSKrzysztof Grobelny if (redfishAction == 303479e899dSKrzysztof Grobelny metric_report_definition::ReportActionsEnum::Invalid) 304479e899dSKrzysztof Grobelny { 305479e899dSKrzysztof Grobelny messages::internalError(asyncResp->res); 306479e899dSKrzysztof Grobelny return; 307081ebf06SWludzik, Jozef } 308081ebf06SWludzik, Jozef 309479e899dSKrzysztof Grobelny redfishReportActions.emplace_back(redfishAction); 310479e899dSKrzysztof Grobelny } 311479e899dSKrzysztof Grobelny 312479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["ReportActions"] = std::move(redfishReportActions); 313479e899dSKrzysztof Grobelny 314479e899dSKrzysztof Grobelny nlohmann::json::array_t metrics = nlohmann::json::array(); 315479e899dSKrzysztof Grobelny for (const auto& [sensorData, collectionFunction, collectionTimeScope, 316479e899dSKrzysztof Grobelny collectionDuration] : readingParams) 31789474494SKrzysztof Grobelny { 318479e899dSKrzysztof Grobelny nlohmann::json::array_t metricProperties; 319479e899dSKrzysztof Grobelny 320479e899dSKrzysztof Grobelny for (const auto& [sensorPath, sensorMetadata] : sensorData) 321081ebf06SWludzik, Jozef { 322479e899dSKrzysztof Grobelny metricProperties.emplace_back(sensorMetadata); 323479e899dSKrzysztof Grobelny } 324479e899dSKrzysztof Grobelny 325613dabeaSEd Tanous nlohmann::json::object_t metric; 326479e899dSKrzysztof Grobelny 327479e899dSKrzysztof Grobelny metric_report_definition::CalculationAlgorithmEnum 328479e899dSKrzysztof Grobelny redfishCollectionFunction = 329479e899dSKrzysztof Grobelny telemetry::toRedfishCollectionFunction(collectionFunction); 330479e899dSKrzysztof Grobelny if (redfishCollectionFunction == 331479e899dSKrzysztof Grobelny metric_report_definition::CalculationAlgorithmEnum::Invalid) 332479e899dSKrzysztof Grobelny { 333479e899dSKrzysztof Grobelny messages::internalError(asyncResp->res); 334479e899dSKrzysztof Grobelny return; 335479e899dSKrzysztof Grobelny } 336479e899dSKrzysztof Grobelny metric["CollectionFunction"] = redfishCollectionFunction; 337479e899dSKrzysztof Grobelny 338479e899dSKrzysztof Grobelny metric_report_definition::CollectionTimeScope 339479e899dSKrzysztof Grobelny redfishCollectionTimeScope = 340479e899dSKrzysztof Grobelny toRedfishCollectionTimeScope(collectionTimeScope); 341479e899dSKrzysztof Grobelny if (redfishCollectionTimeScope == 342479e899dSKrzysztof Grobelny metric_report_definition::CollectionTimeScope::Invalid) 343479e899dSKrzysztof Grobelny { 344479e899dSKrzysztof Grobelny messages::internalError(asyncResp->res); 345479e899dSKrzysztof Grobelny return; 346479e899dSKrzysztof Grobelny } 347479e899dSKrzysztof Grobelny metric["CollectionTimeScope"] = redfishCollectionTimeScope; 348479e899dSKrzysztof Grobelny 349479e899dSKrzysztof Grobelny metric["MetricProperties"] = std::move(metricProperties); 350479e899dSKrzysztof Grobelny metric["CollectionDuration"] = time_utils::toDurationString( 351479e899dSKrzysztof Grobelny std::chrono::milliseconds(collectionDuration)); 352b2ba3072SPatrick Williams metrics.emplace_back(std::move(metric)); 353081ebf06SWludzik, Jozef } 354479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["Metrics"] = std::move(metrics); 355479e899dSKrzysztof Grobelny 356479e899dSKrzysztof Grobelny if (enabled) 357479e899dSKrzysztof Grobelny { 358479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["Status"]["State"] = "Enabled"; 359479e899dSKrzysztof Grobelny } 360479e899dSKrzysztof Grobelny else 361479e899dSKrzysztof Grobelny { 362479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["Status"]["State"] = "Disabled"; 36389474494SKrzysztof Grobelny } 36489474494SKrzysztof Grobelny 365479e899dSKrzysztof Grobelny metric_report_definition::ReportUpdatesEnum redfishReportUpdates = 366479e899dSKrzysztof Grobelny toRedfishReportUpdates(reportUpdates); 367479e899dSKrzysztof Grobelny if (redfishReportUpdates == 368479e899dSKrzysztof Grobelny metric_report_definition::ReportUpdatesEnum::Invalid) 36989474494SKrzysztof Grobelny { 370479e899dSKrzysztof Grobelny messages::internalError(asyncResp->res); 371479e899dSKrzysztof Grobelny return; 37289474494SKrzysztof Grobelny } 373479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["ReportUpdates"] = redfishReportUpdates; 37489474494SKrzysztof Grobelny 375479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["MetricReportDefinitionEnabled"] = enabled; 376479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["AppendLimit"] = appendLimit; 377479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["Name"] = name; 378081ebf06SWludzik, Jozef asyncResp->res.jsonValue["Schedule"]["RecurrenceInterval"] = 379479e899dSKrzysztof Grobelny time_utils::toDurationString(std::chrono::milliseconds(interval)); 380479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["@odata.type"] = 381479e899dSKrzysztof Grobelny "#MetricReportDefinition.v1_3_0.MetricReportDefinition"; 382479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["@odata.id"] = boost::urls::format( 383479e899dSKrzysztof Grobelny "/redfish/v1/TelemetryService/MetricReportDefinitions/{}", id); 384479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["Id"] = id; 385479e899dSKrzysztof Grobelny asyncResp->res.jsonValue["MetricReport"]["@odata.id"] = boost::urls::format( 386479e899dSKrzysztof Grobelny "/redfish/v1/TelemetryService/MetricReports/{}", id); 38789474494SKrzysztof Grobelny } 38889474494SKrzysztof Grobelny 3894dbb8aeaSWludzik, Jozef struct AddReportArgs 3904dbb8aeaSWludzik, Jozef { 391479e899dSKrzysztof Grobelny struct MetricArgs 392479e899dSKrzysztof Grobelny { 393479e899dSKrzysztof Grobelny std::vector<std::string> uris; 394479e899dSKrzysztof Grobelny std::string collectionFunction; 395479e899dSKrzysztof Grobelny std::string collectionTimeScope; 396479e899dSKrzysztof Grobelny uint64_t collectionDuration = 0; 397479e899dSKrzysztof Grobelny }; 398479e899dSKrzysztof Grobelny 399479e899dSKrzysztof Grobelny std::string id; 4004dbb8aeaSWludzik, Jozef std::string name; 4014dbb8aeaSWludzik, Jozef std::string reportingType; 402479e899dSKrzysztof Grobelny std::string reportUpdates; 403479e899dSKrzysztof Grobelny uint64_t appendLimit = std::numeric_limits<uint64_t>::max(); 404479e899dSKrzysztof Grobelny std::vector<std::string> reportActions; 405479e899dSKrzysztof Grobelny uint64_t interval = std::numeric_limits<uint64_t>::max(); 406479e899dSKrzysztof Grobelny std::vector<MetricArgs> metrics; 407479e899dSKrzysztof Grobelny bool metricReportDefinitionEnabled = true; 4084dbb8aeaSWludzik, Jozef }; 4094dbb8aeaSWludzik, Jozef 4104dbb8aeaSWludzik, Jozef inline bool toDbusReportActions(crow::Response& res, 411479e899dSKrzysztof Grobelny const std::vector<std::string>& actions, 4129e6c388aSLukasz Kazmierczak std::vector<std::string>& outReportActions) 4134dbb8aeaSWludzik, Jozef { 4144dbb8aeaSWludzik, Jozef size_t index = 0; 415479e899dSKrzysztof Grobelny for (const std::string& action : actions) 4164dbb8aeaSWludzik, Jozef { 417479e899dSKrzysztof Grobelny std::string dbusReportAction = toDbusReportAction(action); 418479e899dSKrzysztof Grobelny if (dbusReportAction.empty()) 4194dbb8aeaSWludzik, Jozef { 4209e6c388aSLukasz Kazmierczak messages::propertyValueNotInList( 4219e6c388aSLukasz Kazmierczak res, action, "ReportActions/" + std::to_string(index)); 4224dbb8aeaSWludzik, Jozef return false; 4234dbb8aeaSWludzik, Jozef } 424479e899dSKrzysztof Grobelny 4259e6c388aSLukasz Kazmierczak outReportActions.emplace_back(std::move(dbusReportAction)); 4264dbb8aeaSWludzik, Jozef index++; 4274dbb8aeaSWludzik, Jozef } 4284dbb8aeaSWludzik, Jozef return true; 4294dbb8aeaSWludzik, Jozef } 4304dbb8aeaSWludzik, Jozef 431b14f357fSEd Tanous inline bool getUserMetric(crow::Response& res, nlohmann::json::object_t& metric, 432479e899dSKrzysztof Grobelny AddReportArgs::MetricArgs& metricArgs) 433479e899dSKrzysztof Grobelny { 434479e899dSKrzysztof Grobelny std::optional<std::vector<std::string>> uris; 435479e899dSKrzysztof Grobelny std::optional<std::string> collectionDurationStr; 436479e899dSKrzysztof Grobelny std::optional<std::string> collectionFunction; 437479e899dSKrzysztof Grobelny std::optional<std::string> collectionTimeScopeStr; 438479e899dSKrzysztof Grobelny 439b14f357fSEd Tanous if (!json_util::readJsonObject( 440b14f357fSEd Tanous metric, res, "MetricProperties", uris, "CollectionFunction", 441b14f357fSEd Tanous collectionFunction, "CollectionTimeScope", collectionTimeScopeStr, 442479e899dSKrzysztof Grobelny "CollectionDuration", collectionDurationStr)) 443479e899dSKrzysztof Grobelny { 444479e899dSKrzysztof Grobelny return false; 445479e899dSKrzysztof Grobelny } 446479e899dSKrzysztof Grobelny 447479e899dSKrzysztof Grobelny if (uris) 448479e899dSKrzysztof Grobelny { 449479e899dSKrzysztof Grobelny metricArgs.uris = std::move(*uris); 450479e899dSKrzysztof Grobelny } 451479e899dSKrzysztof Grobelny 452479e899dSKrzysztof Grobelny if (collectionFunction) 453479e899dSKrzysztof Grobelny { 454479e899dSKrzysztof Grobelny std::string dbusCollectionFunction = 455479e899dSKrzysztof Grobelny telemetry::toDbusCollectionFunction(*collectionFunction); 456479e899dSKrzysztof Grobelny if (dbusCollectionFunction.empty()) 457479e899dSKrzysztof Grobelny { 458479e899dSKrzysztof Grobelny messages::propertyValueIncorrect(res, "CollectionFunction", 459479e899dSKrzysztof Grobelny *collectionFunction); 460479e899dSKrzysztof Grobelny return false; 461479e899dSKrzysztof Grobelny } 462479e899dSKrzysztof Grobelny metricArgs.collectionFunction = std::move(dbusCollectionFunction); 463479e899dSKrzysztof Grobelny } 464479e899dSKrzysztof Grobelny 465479e899dSKrzysztof Grobelny if (collectionTimeScopeStr) 466479e899dSKrzysztof Grobelny { 467479e899dSKrzysztof Grobelny std::string dbusCollectionTimeScope = 468479e899dSKrzysztof Grobelny toDbusCollectionTimeScope(*collectionTimeScopeStr); 469479e899dSKrzysztof Grobelny if (dbusCollectionTimeScope.empty()) 470479e899dSKrzysztof Grobelny { 471479e899dSKrzysztof Grobelny messages::propertyValueIncorrect(res, "CollectionTimeScope", 472479e899dSKrzysztof Grobelny *collectionTimeScopeStr); 473479e899dSKrzysztof Grobelny return false; 474479e899dSKrzysztof Grobelny } 475479e899dSKrzysztof Grobelny metricArgs.collectionTimeScope = std::move(dbusCollectionTimeScope); 476479e899dSKrzysztof Grobelny } 477479e899dSKrzysztof Grobelny 478479e899dSKrzysztof Grobelny if (collectionDurationStr) 479479e899dSKrzysztof Grobelny { 480479e899dSKrzysztof Grobelny std::optional<std::chrono::milliseconds> duration = 481479e899dSKrzysztof Grobelny time_utils::fromDurationString(*collectionDurationStr); 482479e899dSKrzysztof Grobelny 483479e899dSKrzysztof Grobelny if (!duration || duration->count() < 0) 484479e899dSKrzysztof Grobelny { 485479e899dSKrzysztof Grobelny messages::propertyValueIncorrect(res, "CollectionDuration", 486479e899dSKrzysztof Grobelny *collectionDurationStr); 487479e899dSKrzysztof Grobelny return false; 488479e899dSKrzysztof Grobelny } 489479e899dSKrzysztof Grobelny 490479e899dSKrzysztof Grobelny metricArgs.collectionDuration = 491479e899dSKrzysztof Grobelny static_cast<uint64_t>(duration->count()); 492479e899dSKrzysztof Grobelny } 493479e899dSKrzysztof Grobelny 494479e899dSKrzysztof Grobelny return true; 495479e899dSKrzysztof Grobelny } 496479e899dSKrzysztof Grobelny 497479e899dSKrzysztof Grobelny inline bool getUserMetrics(crow::Response& res, 498b14f357fSEd Tanous std::span<nlohmann::json::object_t> metrics, 499479e899dSKrzysztof Grobelny std::vector<AddReportArgs::MetricArgs>& result) 500479e899dSKrzysztof Grobelny { 501479e899dSKrzysztof Grobelny result.reserve(metrics.size()); 502479e899dSKrzysztof Grobelny 503b14f357fSEd Tanous for (nlohmann::json::object_t& m : metrics) 504479e899dSKrzysztof Grobelny { 505479e899dSKrzysztof Grobelny AddReportArgs::MetricArgs metricArgs; 506479e899dSKrzysztof Grobelny 507479e899dSKrzysztof Grobelny if (!getUserMetric(res, m, metricArgs)) 508479e899dSKrzysztof Grobelny { 509479e899dSKrzysztof Grobelny return false; 510479e899dSKrzysztof Grobelny } 511479e899dSKrzysztof Grobelny 512479e899dSKrzysztof Grobelny result.emplace_back(std::move(metricArgs)); 513479e899dSKrzysztof Grobelny } 514479e899dSKrzysztof Grobelny 515479e899dSKrzysztof Grobelny return true; 516479e899dSKrzysztof Grobelny } 517479e899dSKrzysztof Grobelny 5184dbb8aeaSWludzik, Jozef inline bool getUserParameters(crow::Response& res, const crow::Request& req, 5194dbb8aeaSWludzik, Jozef AddReportArgs& args) 5204dbb8aeaSWludzik, Jozef { 521479e899dSKrzysztof Grobelny std::optional<std::string> id; 522479e899dSKrzysztof Grobelny std::optional<std::string> name; 523479e899dSKrzysztof Grobelny std::optional<std::string> reportingTypeStr; 524479e899dSKrzysztof Grobelny std::optional<std::string> reportUpdatesStr; 525479e899dSKrzysztof Grobelny std::optional<uint64_t> appendLimit; 526479e899dSKrzysztof Grobelny std::optional<bool> metricReportDefinitionEnabled; 527b14f357fSEd Tanous std::optional<std::vector<nlohmann::json::object_t>> metrics; 528479e899dSKrzysztof Grobelny std::optional<std::vector<std::string>> reportActionsStr; 529b14f357fSEd Tanous std::optional<std::string> scheduleDurationStr; 530479e899dSKrzysztof Grobelny 531479e899dSKrzysztof Grobelny if (!json_util::readJsonPatch( 532479e899dSKrzysztof Grobelny req, res, "Id", id, "Name", name, "Metrics", metrics, 533479e899dSKrzysztof Grobelny "MetricReportDefinitionType", reportingTypeStr, "ReportUpdates", 534479e899dSKrzysztof Grobelny reportUpdatesStr, "AppendLimit", appendLimit, "ReportActions", 535b14f357fSEd Tanous reportActionsStr, "Schedule/RecurrenceInterval", 536b14f357fSEd Tanous scheduleDurationStr, "MetricReportDefinitionEnabled", 537b14f357fSEd Tanous metricReportDefinitionEnabled)) 5384dbb8aeaSWludzik, Jozef { 5394dbb8aeaSWludzik, Jozef return false; 5404dbb8aeaSWludzik, Jozef } 5414dbb8aeaSWludzik, Jozef 542479e899dSKrzysztof Grobelny if (id) 543479e899dSKrzysztof Grobelny { 544479e899dSKrzysztof Grobelny constexpr const char* allowedCharactersInId = 5454dbb8aeaSWludzik, Jozef "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789_"; 546479e899dSKrzysztof Grobelny if (id->empty() || 547479e899dSKrzysztof Grobelny id->find_first_not_of(allowedCharactersInId) != std::string::npos) 5484dbb8aeaSWludzik, Jozef { 549479e899dSKrzysztof Grobelny messages::propertyValueIncorrect(res, "Id", *id); 5504dbb8aeaSWludzik, Jozef return false; 5514dbb8aeaSWludzik, Jozef } 552479e899dSKrzysztof Grobelny args.id = *id; 553479e899dSKrzysztof Grobelny } 5544dbb8aeaSWludzik, Jozef 555479e899dSKrzysztof Grobelny if (name) 5564dbb8aeaSWludzik, Jozef { 557479e899dSKrzysztof Grobelny args.name = *name; 558479e899dSKrzysztof Grobelny } 559479e899dSKrzysztof Grobelny 560479e899dSKrzysztof Grobelny if (reportingTypeStr) 561479e899dSKrzysztof Grobelny { 562479e899dSKrzysztof Grobelny std::string dbusReportingType = toDbusReportingType(*reportingTypeStr); 563479e899dSKrzysztof Grobelny if (dbusReportingType.empty()) 564479e899dSKrzysztof Grobelny { 565479e899dSKrzysztof Grobelny messages::propertyValueNotInList(res, *reportingTypeStr, 5664dbb8aeaSWludzik, Jozef "MetricReportDefinitionType"); 5674dbb8aeaSWludzik, Jozef return false; 5684dbb8aeaSWludzik, Jozef } 569479e899dSKrzysztof Grobelny args.reportingType = dbusReportingType; 570479e899dSKrzysztof Grobelny } 5714dbb8aeaSWludzik, Jozef 572479e899dSKrzysztof Grobelny if (reportUpdatesStr) 573479e899dSKrzysztof Grobelny { 574479e899dSKrzysztof Grobelny std::string dbusReportUpdates = toDbusReportUpdates(*reportUpdatesStr); 575479e899dSKrzysztof Grobelny if (dbusReportUpdates.empty()) 576479e899dSKrzysztof Grobelny { 577479e899dSKrzysztof Grobelny messages::propertyValueNotInList(res, *reportUpdatesStr, 578479e899dSKrzysztof Grobelny "ReportUpdates"); 579479e899dSKrzysztof Grobelny return false; 580479e899dSKrzysztof Grobelny } 581479e899dSKrzysztof Grobelny args.reportUpdates = dbusReportUpdates; 582479e899dSKrzysztof Grobelny } 583479e899dSKrzysztof Grobelny 584479e899dSKrzysztof Grobelny if (appendLimit) 585479e899dSKrzysztof Grobelny { 586479e899dSKrzysztof Grobelny args.appendLimit = *appendLimit; 587479e899dSKrzysztof Grobelny } 588479e899dSKrzysztof Grobelny 589479e899dSKrzysztof Grobelny if (metricReportDefinitionEnabled) 590479e899dSKrzysztof Grobelny { 591479e899dSKrzysztof Grobelny args.metricReportDefinitionEnabled = *metricReportDefinitionEnabled; 592479e899dSKrzysztof Grobelny } 593479e899dSKrzysztof Grobelny 594479e899dSKrzysztof Grobelny if (reportActionsStr) 595479e899dSKrzysztof Grobelny { 5969e6c388aSLukasz Kazmierczak if (!toDbusReportActions(res, *reportActionsStr, args.reportActions)) 5974dbb8aeaSWludzik, Jozef { 5984dbb8aeaSWludzik, Jozef return false; 5994dbb8aeaSWludzik, Jozef } 600479e899dSKrzysztof Grobelny } 6014dbb8aeaSWludzik, Jozef 602479e899dSKrzysztof Grobelny if (reportingTypeStr == "Periodic") 6034dbb8aeaSWludzik, Jozef { 604b14f357fSEd Tanous if (!scheduleDurationStr) 6054dbb8aeaSWludzik, Jozef { 6064dbb8aeaSWludzik, Jozef messages::createFailedMissingReqProperties(res, "Schedule"); 6074dbb8aeaSWludzik, Jozef return false; 6084dbb8aeaSWludzik, Jozef } 6094dbb8aeaSWludzik, Jozef 6104dbb8aeaSWludzik, Jozef std::optional<std::chrono::milliseconds> durationNum = 611b14f357fSEd Tanous time_utils::fromDurationString(*scheduleDurationStr); 612479e899dSKrzysztof Grobelny if (!durationNum || durationNum->count() < 0) 6134dbb8aeaSWludzik, Jozef { 6144dbb8aeaSWludzik, Jozef messages::propertyValueIncorrect(res, "RecurrenceInterval", 615b14f357fSEd Tanous *scheduleDurationStr); 6164dbb8aeaSWludzik, Jozef return false; 6174dbb8aeaSWludzik, Jozef } 6184dbb8aeaSWludzik, Jozef args.interval = static_cast<uint64_t>(durationNum->count()); 6194dbb8aeaSWludzik, Jozef } 6204dbb8aeaSWludzik, Jozef 621479e899dSKrzysztof Grobelny if (metrics) 6224dbb8aeaSWludzik, Jozef { 623479e899dSKrzysztof Grobelny if (!getUserMetrics(res, *metrics, args.metrics)) 6244dbb8aeaSWludzik, Jozef { 6254dbb8aeaSWludzik, Jozef return false; 6264dbb8aeaSWludzik, Jozef } 6274dbb8aeaSWludzik, Jozef } 6284dbb8aeaSWludzik, Jozef 6294dbb8aeaSWludzik, Jozef return true; 6304dbb8aeaSWludzik, Jozef } 6314dbb8aeaSWludzik, Jozef 632ca1600c1SSzymon Dompke inline bool getChassisSensorNodeFromMetrics( 6338d1b46d7Szhanghch05 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 634479e899dSKrzysztof Grobelny std::span<const AddReportArgs::MetricArgs> metrics, 6354dbb8aeaSWludzik, Jozef boost::container::flat_set<std::pair<std::string, std::string>>& matched) 6364dbb8aeaSWludzik, Jozef { 637ca1600c1SSzymon Dompke for (const auto& metric : metrics) 6384dbb8aeaSWludzik, Jozef { 639479e899dSKrzysztof Grobelny std::optional<IncorrectMetricUri> error = 640479e899dSKrzysztof Grobelny getChassisSensorNode(metric.uris, matched); 641ca1600c1SSzymon Dompke if (error) 6424dbb8aeaSWludzik, Jozef { 643ca1600c1SSzymon Dompke messages::propertyValueIncorrect(asyncResp->res, error->uri, 6444dbb8aeaSWludzik, Jozef "MetricProperties/" + 645ca1600c1SSzymon Dompke std::to_string(error->index)); 6464dbb8aeaSWludzik, Jozef return false; 6474dbb8aeaSWludzik, Jozef } 6484dbb8aeaSWludzik, Jozef } 6494dbb8aeaSWludzik, Jozef return true; 6504dbb8aeaSWludzik, Jozef } 6514dbb8aeaSWludzik, Jozef 6524dbb8aeaSWludzik, Jozef class AddReport 6534dbb8aeaSWludzik, Jozef { 6544dbb8aeaSWludzik, Jozef public: 655*891eaa7cSEd Tanous AddReport(AddReportArgs&& argsIn, 6568a592810SEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn) : 6578a592810SEd Tanous asyncResp(asyncRespIn), 6589e6c388aSLukasz Kazmierczak args(std::move(argsIn)) 6594dbb8aeaSWludzik, Jozef {} 660479e899dSKrzysztof Grobelny 6614dbb8aeaSWludzik, Jozef ~AddReport() 6624dbb8aeaSWludzik, Jozef { 663479e899dSKrzysztof Grobelny boost::asio::post(crow::connections::systemBus->get_io_context(), 664479e899dSKrzysztof Grobelny std::bind_front(&performAddReport, asyncResp, args, 665479e899dSKrzysztof Grobelny std::move(uriToDbus))); 666479e899dSKrzysztof Grobelny } 667479e899dSKrzysztof Grobelny 668479e899dSKrzysztof Grobelny static void performAddReport( 669479e899dSKrzysztof Grobelny const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 670479e899dSKrzysztof Grobelny const AddReportArgs& args, 671479e899dSKrzysztof Grobelny const boost::container::flat_map<std::string, std::string>& uriToDbus) 672479e899dSKrzysztof Grobelny { 6734dbb8aeaSWludzik, Jozef if (asyncResp->res.result() != boost::beast::http::status::ok) 6744dbb8aeaSWludzik, Jozef { 6754dbb8aeaSWludzik, Jozef return; 6764dbb8aeaSWludzik, Jozef } 6774dbb8aeaSWludzik, Jozef 6784dbb8aeaSWludzik, Jozef telemetry::ReadingParameters readingParams; 6794dbb8aeaSWludzik, Jozef readingParams.reserve(args.metrics.size()); 6804dbb8aeaSWludzik, Jozef 681479e899dSKrzysztof Grobelny for (const auto& metric : args.metrics) 6824dbb8aeaSWludzik, Jozef { 683479e899dSKrzysztof Grobelny std::vector< 684479e899dSKrzysztof Grobelny std::tuple<sdbusplus::message::object_path, std::string>> 685479e899dSKrzysztof Grobelny sensorParams; 686479e899dSKrzysztof Grobelny sensorParams.reserve(metric.uris.size()); 687479e899dSKrzysztof Grobelny 688479e899dSKrzysztof Grobelny for (size_t i = 0; i < metric.uris.size(); i++) 6894dbb8aeaSWludzik, Jozef { 690479e899dSKrzysztof Grobelny const std::string& uri = metric.uris[i]; 6914dbb8aeaSWludzik, Jozef auto el = uriToDbus.find(uri); 6924dbb8aeaSWludzik, Jozef if (el == uriToDbus.end()) 6934dbb8aeaSWludzik, Jozef { 69462598e31SEd Tanous BMCWEB_LOG_ERROR( 69562598e31SEd Tanous "Failed to find DBus sensor corresponding to URI {}", 69662598e31SEd Tanous uri); 6974dbb8aeaSWludzik, Jozef messages::propertyValueNotInList(asyncResp->res, uri, 6984dbb8aeaSWludzik, Jozef "MetricProperties/" + 6994dbb8aeaSWludzik, Jozef std::to_string(i)); 7004dbb8aeaSWludzik, Jozef return; 7014dbb8aeaSWludzik, Jozef } 7024dbb8aeaSWludzik, Jozef 7034dbb8aeaSWludzik, Jozef const std::string& dbusPath = el->second; 704479e899dSKrzysztof Grobelny sensorParams.emplace_back(dbusPath, uri); 7054dbb8aeaSWludzik, Jozef } 706479e899dSKrzysztof Grobelny 707479e899dSKrzysztof Grobelny readingParams.emplace_back( 708479e899dSKrzysztof Grobelny std::move(sensorParams), metric.collectionFunction, 709479e899dSKrzysztof Grobelny metric.collectionTimeScope, metric.collectionDuration); 7104dbb8aeaSWludzik, Jozef } 711479e899dSKrzysztof Grobelny 7124dbb8aeaSWludzik, Jozef crow::connections::systemBus->async_method_call( 713479e899dSKrzysztof Grobelny [asyncResp, id = args.id, uriToDbus]( 7145e7e2dc5SEd Tanous const boost::system::error_code& ec, const std::string&) { 7154dbb8aeaSWludzik, Jozef if (ec == boost::system::errc::file_exists) 7164dbb8aeaSWludzik, Jozef { 7174dbb8aeaSWludzik, Jozef messages::resourceAlreadyExists( 718479e899dSKrzysztof Grobelny asyncResp->res, "MetricReportDefinition", "Id", id); 7194dbb8aeaSWludzik, Jozef return; 7204dbb8aeaSWludzik, Jozef } 7214dbb8aeaSWludzik, Jozef if (ec == boost::system::errc::too_many_files_open) 7224dbb8aeaSWludzik, Jozef { 723479e899dSKrzysztof Grobelny messages::createLimitReachedForResource(asyncResp->res); 7244dbb8aeaSWludzik, Jozef return; 7254dbb8aeaSWludzik, Jozef } 7264dbb8aeaSWludzik, Jozef if (ec == boost::system::errc::argument_list_too_long) 7274dbb8aeaSWludzik, Jozef { 7284dbb8aeaSWludzik, Jozef nlohmann::json metricProperties = nlohmann::json::array(); 7294dbb8aeaSWludzik, Jozef for (const auto& [uri, _] : uriToDbus) 7304dbb8aeaSWludzik, Jozef { 7314dbb8aeaSWludzik, Jozef metricProperties.emplace_back(uri); 7324dbb8aeaSWludzik, Jozef } 73314fbced6SEd Tanous messages::propertyValueIncorrect( 734367b3dceSGinu George asyncResp->res, "MetricProperties", metricProperties); 7354dbb8aeaSWludzik, Jozef return; 7364dbb8aeaSWludzik, Jozef } 7374dbb8aeaSWludzik, Jozef if (ec) 7384dbb8aeaSWludzik, Jozef { 739479e899dSKrzysztof Grobelny messages::internalError(asyncResp->res); 74062598e31SEd Tanous BMCWEB_LOG_ERROR("respHandler DBus error {}", ec); 7414dbb8aeaSWludzik, Jozef return; 7424dbb8aeaSWludzik, Jozef } 7434dbb8aeaSWludzik, Jozef 744479e899dSKrzysztof Grobelny messages::created(asyncResp->res); 7454dbb8aeaSWludzik, Jozef }, 7464dbb8aeaSWludzik, Jozef telemetry::service, "/xyz/openbmc_project/Telemetry/Reports", 7474dbb8aeaSWludzik, Jozef "xyz.openbmc_project.Telemetry.ReportManager", "AddReport", 748479e899dSKrzysztof Grobelny "TelemetryService/" + args.id, args.name, args.reportingType, 749479e899dSKrzysztof Grobelny args.reportUpdates, args.appendLimit, args.reportActions, 750479e899dSKrzysztof Grobelny args.interval, readingParams, args.metricReportDefinitionEnabled); 7514dbb8aeaSWludzik, Jozef } 7524dbb8aeaSWludzik, Jozef 753ecd6a3a2SEd Tanous AddReport(const AddReport&) = delete; 754ecd6a3a2SEd Tanous AddReport(AddReport&&) = delete; 755ecd6a3a2SEd Tanous AddReport& operator=(const AddReport&) = delete; 756ecd6a3a2SEd Tanous AddReport& operator=(AddReport&&) = delete; 757ecd6a3a2SEd Tanous 758fe04d49cSNan Zhou void insert(const std::map<std::string, std::string>& el) 7594dbb8aeaSWludzik, Jozef { 7604dbb8aeaSWludzik, Jozef uriToDbus.insert(el.begin(), el.end()); 7614dbb8aeaSWludzik, Jozef } 7624dbb8aeaSWludzik, Jozef 7634dbb8aeaSWludzik, Jozef private: 764479e899dSKrzysztof Grobelny std::shared_ptr<bmcweb::AsyncResp> asyncResp; 7654dbb8aeaSWludzik, Jozef AddReportArgs args; 76647f2934cSEd Tanous boost::container::flat_map<std::string, std::string> uriToDbus; 7674dbb8aeaSWludzik, Jozef }; 7689e6c388aSLukasz Kazmierczak 7699e6c388aSLukasz Kazmierczak class UpdateMetrics 7709e6c388aSLukasz Kazmierczak { 7719e6c388aSLukasz Kazmierczak public: 7729e6c388aSLukasz Kazmierczak UpdateMetrics(std::string_view idIn, 7739e6c388aSLukasz Kazmierczak const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn) : 7749e6c388aSLukasz Kazmierczak id(idIn), 7759e6c388aSLukasz Kazmierczak asyncResp(asyncRespIn) 7769e6c388aSLukasz Kazmierczak {} 7779e6c388aSLukasz Kazmierczak 7789e6c388aSLukasz Kazmierczak ~UpdateMetrics() 7799e6c388aSLukasz Kazmierczak { 7809e6c388aSLukasz Kazmierczak try 7819e6c388aSLukasz Kazmierczak { 7829e6c388aSLukasz Kazmierczak setReadingParams(); 7839e6c388aSLukasz Kazmierczak } 7849e6c388aSLukasz Kazmierczak catch (const std::exception& e) 7859e6c388aSLukasz Kazmierczak { 7869e6c388aSLukasz Kazmierczak BMCWEB_LOG_ERROR("{}", e.what()); 7879e6c388aSLukasz Kazmierczak } 7889e6c388aSLukasz Kazmierczak catch (...) 7899e6c388aSLukasz Kazmierczak { 7909e6c388aSLukasz Kazmierczak BMCWEB_LOG_ERROR("Unknown error"); 7919e6c388aSLukasz Kazmierczak } 7929e6c388aSLukasz Kazmierczak } 7939e6c388aSLukasz Kazmierczak 7949e6c388aSLukasz Kazmierczak UpdateMetrics(const UpdateMetrics&) = delete; 7959e6c388aSLukasz Kazmierczak UpdateMetrics(UpdateMetrics&&) = delete; 7969e6c388aSLukasz Kazmierczak UpdateMetrics& operator=(const UpdateMetrics&) = delete; 7979e6c388aSLukasz Kazmierczak UpdateMetrics& operator=(UpdateMetrics&&) = delete; 7989e6c388aSLukasz Kazmierczak 7999e6c388aSLukasz Kazmierczak std::string id; 8009e6c388aSLukasz Kazmierczak std::map<std::string, std::string> metricPropertyToDbusPaths; 8019e6c388aSLukasz Kazmierczak 8029e6c388aSLukasz Kazmierczak void insert(const std::map<std::string, std::string>& 8039e6c388aSLukasz Kazmierczak additionalMetricPropertyToDbusPaths) 8049e6c388aSLukasz Kazmierczak { 8059e6c388aSLukasz Kazmierczak metricPropertyToDbusPaths.insert( 8069e6c388aSLukasz Kazmierczak additionalMetricPropertyToDbusPaths.begin(), 8079e6c388aSLukasz Kazmierczak additionalMetricPropertyToDbusPaths.end()); 8089e6c388aSLukasz Kazmierczak } 8099e6c388aSLukasz Kazmierczak 8109e6c388aSLukasz Kazmierczak void emplace(std::span<const std::tuple<sdbusplus::message::object_path, 8119e6c388aSLukasz Kazmierczak std::string>> 8129e6c388aSLukasz Kazmierczak pathAndUri, 8139e6c388aSLukasz Kazmierczak const AddReportArgs::MetricArgs& metricArgs) 8149e6c388aSLukasz Kazmierczak { 8159e6c388aSLukasz Kazmierczak readingParamsUris.emplace_back(metricArgs.uris); 8169e6c388aSLukasz Kazmierczak readingParams.emplace_back( 8179e6c388aSLukasz Kazmierczak std::vector(pathAndUri.begin(), pathAndUri.end()), 8189e6c388aSLukasz Kazmierczak metricArgs.collectionFunction, metricArgs.collectionTimeScope, 8199e6c388aSLukasz Kazmierczak metricArgs.collectionDuration); 8209e6c388aSLukasz Kazmierczak } 8219e6c388aSLukasz Kazmierczak 8229e6c388aSLukasz Kazmierczak void setReadingParams() 8239e6c388aSLukasz Kazmierczak { 8249e6c388aSLukasz Kazmierczak if (asyncResp->res.result() != boost::beast::http::status::ok) 8259e6c388aSLukasz Kazmierczak { 8269e6c388aSLukasz Kazmierczak return; 8279e6c388aSLukasz Kazmierczak } 8289e6c388aSLukasz Kazmierczak 8299e6c388aSLukasz Kazmierczak for (size_t index = 0; index < readingParamsUris.size(); ++index) 8309e6c388aSLukasz Kazmierczak { 8319e6c388aSLukasz Kazmierczak std::span<const std::string> newUris = readingParamsUris[index]; 8329e6c388aSLukasz Kazmierczak 8339e6c388aSLukasz Kazmierczak const std::optional<std::vector< 8349e6c388aSLukasz Kazmierczak std::tuple<sdbusplus::message::object_path, std::string>>> 8359e6c388aSLukasz Kazmierczak readingParam = sensorPathToUri(newUris); 8369e6c388aSLukasz Kazmierczak 8379e6c388aSLukasz Kazmierczak if (!readingParam) 8389e6c388aSLukasz Kazmierczak { 8399e6c388aSLukasz Kazmierczak return; 8409e6c388aSLukasz Kazmierczak } 8419e6c388aSLukasz Kazmierczak 8429e6c388aSLukasz Kazmierczak std::get<0>(readingParams[index]) = *readingParam; 8439e6c388aSLukasz Kazmierczak } 8449e6c388aSLukasz Kazmierczak 8459e6c388aSLukasz Kazmierczak crow::connections::systemBus->async_method_call( 8469e6c388aSLukasz Kazmierczak [asyncResp(this->asyncResp), 8479e6c388aSLukasz Kazmierczak reportId = id](const boost::system::error_code& ec) { 8489e6c388aSLukasz Kazmierczak if (!verifyCommonErrors(asyncResp->res, reportId, ec)) 8499e6c388aSLukasz Kazmierczak { 8509e6c388aSLukasz Kazmierczak return; 8519e6c388aSLukasz Kazmierczak } 8529e6c388aSLukasz Kazmierczak }, 8539e6c388aSLukasz Kazmierczak "xyz.openbmc_project.Telemetry", getDbusReportPath(id), 8549e6c388aSLukasz Kazmierczak "org.freedesktop.DBus.Properties", "Set", 8559e6c388aSLukasz Kazmierczak "xyz.openbmc_project.Telemetry.Report", "ReadingParameters", 8569e6c388aSLukasz Kazmierczak dbus::utility::DbusVariantType{readingParams}); 8579e6c388aSLukasz Kazmierczak } 8589e6c388aSLukasz Kazmierczak 8599e6c388aSLukasz Kazmierczak private: 8609e6c388aSLukasz Kazmierczak std::optional< 8619e6c388aSLukasz Kazmierczak std::vector<std::tuple<sdbusplus::message::object_path, std::string>>> 8629e6c388aSLukasz Kazmierczak sensorPathToUri(std::span<const std::string> uris) const 8639e6c388aSLukasz Kazmierczak { 8649e6c388aSLukasz Kazmierczak std::vector<std::tuple<sdbusplus::message::object_path, std::string>> 8659e6c388aSLukasz Kazmierczak result; 8669e6c388aSLukasz Kazmierczak 8679e6c388aSLukasz Kazmierczak for (const std::string& uri : uris) 8689e6c388aSLukasz Kazmierczak { 8699e6c388aSLukasz Kazmierczak auto it = metricPropertyToDbusPaths.find(uri); 8709e6c388aSLukasz Kazmierczak if (it == metricPropertyToDbusPaths.end()) 8719e6c388aSLukasz Kazmierczak { 8729e6c388aSLukasz Kazmierczak messages::propertyValueNotInList(asyncResp->res, uri, 8739e6c388aSLukasz Kazmierczak "MetricProperties"); 8749e6c388aSLukasz Kazmierczak return {}; 8759e6c388aSLukasz Kazmierczak } 8769e6c388aSLukasz Kazmierczak result.emplace_back(it->second, uri); 8779e6c388aSLukasz Kazmierczak } 8789e6c388aSLukasz Kazmierczak 8799e6c388aSLukasz Kazmierczak return result; 8809e6c388aSLukasz Kazmierczak } 8819e6c388aSLukasz Kazmierczak 8829e6c388aSLukasz Kazmierczak const std::shared_ptr<bmcweb::AsyncResp> asyncResp; 8839e6c388aSLukasz Kazmierczak std::vector<std::vector<std::string>> readingParamsUris; 88447f2934cSEd Tanous ReadingParameters readingParams; 8859e6c388aSLukasz Kazmierczak }; 8869e6c388aSLukasz Kazmierczak 8879e6c388aSLukasz Kazmierczak inline void 8889e6c388aSLukasz Kazmierczak setReportEnabled(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 8899e6c388aSLukasz Kazmierczak std::string_view id, bool enabled) 8909e6c388aSLukasz Kazmierczak { 8919e6c388aSLukasz Kazmierczak crow::connections::systemBus->async_method_call( 8929e6c388aSLukasz Kazmierczak [asyncResp, id = std::string(id)](const boost::system::error_code& ec) { 8939e6c388aSLukasz Kazmierczak if (!verifyCommonErrors(asyncResp->res, id, ec)) 8949e6c388aSLukasz Kazmierczak { 8959e6c388aSLukasz Kazmierczak return; 8969e6c388aSLukasz Kazmierczak } 8979e6c388aSLukasz Kazmierczak }, 8989e6c388aSLukasz Kazmierczak "xyz.openbmc_project.Telemetry", getDbusReportPath(id), 8999e6c388aSLukasz Kazmierczak "org.freedesktop.DBus.Properties", "Set", 9009e6c388aSLukasz Kazmierczak "xyz.openbmc_project.Telemetry.Report", "Enabled", 9019e6c388aSLukasz Kazmierczak dbus::utility::DbusVariantType{enabled}); 9029e6c388aSLukasz Kazmierczak } 9039e6c388aSLukasz Kazmierczak 9049e6c388aSLukasz Kazmierczak inline void setReportTypeAndInterval( 9059e6c388aSLukasz Kazmierczak const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, std::string_view id, 9069e6c388aSLukasz Kazmierczak const std::string& reportingType, uint64_t recurrenceInterval) 9079e6c388aSLukasz Kazmierczak { 9089e6c388aSLukasz Kazmierczak crow::connections::systemBus->async_method_call( 9099e6c388aSLukasz Kazmierczak [asyncResp, id = std::string(id)](const boost::system::error_code& ec) { 9109e6c388aSLukasz Kazmierczak if (!verifyCommonErrors(asyncResp->res, id, ec)) 9119e6c388aSLukasz Kazmierczak { 9129e6c388aSLukasz Kazmierczak return; 9139e6c388aSLukasz Kazmierczak } 9149e6c388aSLukasz Kazmierczak }, 9159e6c388aSLukasz Kazmierczak "xyz.openbmc_project.Telemetry", getDbusReportPath(id), 9169e6c388aSLukasz Kazmierczak "xyz.openbmc_project.Telemetry.Report", "SetReportingProperties", 9179e6c388aSLukasz Kazmierczak reportingType, recurrenceInterval); 9189e6c388aSLukasz Kazmierczak } 9199e6c388aSLukasz Kazmierczak 9209e6c388aSLukasz Kazmierczak inline void 9219e6c388aSLukasz Kazmierczak setReportUpdates(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 9229e6c388aSLukasz Kazmierczak std::string_view id, const std::string& reportUpdates) 9239e6c388aSLukasz Kazmierczak { 9249e6c388aSLukasz Kazmierczak crow::connections::systemBus->async_method_call( 9259e6c388aSLukasz Kazmierczak [asyncResp, id = std::string(id)](const boost::system::error_code& ec) { 9269e6c388aSLukasz Kazmierczak if (!verifyCommonErrors(asyncResp->res, id, ec)) 9279e6c388aSLukasz Kazmierczak { 9289e6c388aSLukasz Kazmierczak return; 9299e6c388aSLukasz Kazmierczak } 9309e6c388aSLukasz Kazmierczak }, 9319e6c388aSLukasz Kazmierczak "xyz.openbmc_project.Telemetry", getDbusReportPath(id), 9329e6c388aSLukasz Kazmierczak "org.freedesktop.DBus.Properties", "Set", 9339e6c388aSLukasz Kazmierczak "xyz.openbmc_project.Telemetry.Report", "ReportUpdates", 9349e6c388aSLukasz Kazmierczak dbus::utility::DbusVariantType{reportUpdates}); 9359e6c388aSLukasz Kazmierczak } 9369e6c388aSLukasz Kazmierczak 9379e6c388aSLukasz Kazmierczak inline void 9389e6c388aSLukasz Kazmierczak setReportActions(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 9399e6c388aSLukasz Kazmierczak std::string_view id, 9409e6c388aSLukasz Kazmierczak const std::vector<std::string>& dbusReportActions) 9419e6c388aSLukasz Kazmierczak { 9429e6c388aSLukasz Kazmierczak crow::connections::systemBus->async_method_call( 9439e6c388aSLukasz Kazmierczak [asyncResp, id = std::string(id)](const boost::system::error_code& ec) { 9449e6c388aSLukasz Kazmierczak if (!verifyCommonErrors(asyncResp->res, id, ec)) 9459e6c388aSLukasz Kazmierczak { 9469e6c388aSLukasz Kazmierczak return; 9479e6c388aSLukasz Kazmierczak } 9489e6c388aSLukasz Kazmierczak }, 9499e6c388aSLukasz Kazmierczak "xyz.openbmc_project.Telemetry", getDbusReportPath(id), 9509e6c388aSLukasz Kazmierczak "org.freedesktop.DBus.Properties", "Set", 9519e6c388aSLukasz Kazmierczak "xyz.openbmc_project.Telemetry.Report", "ReportActions", 9529e6c388aSLukasz Kazmierczak dbus::utility::DbusVariantType{dbusReportActions}); 9539e6c388aSLukasz Kazmierczak } 9549e6c388aSLukasz Kazmierczak 9559e6c388aSLukasz Kazmierczak inline void 9569e6c388aSLukasz Kazmierczak setReportMetrics(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 957b14f357fSEd Tanous std::string_view id, 958b14f357fSEd Tanous std::span<nlohmann::json::object_t> metrics) 9599e6c388aSLukasz Kazmierczak { 9609e6c388aSLukasz Kazmierczak sdbusplus::asio::getAllProperties( 9619e6c388aSLukasz Kazmierczak *crow::connections::systemBus, telemetry::service, 9629e6c388aSLukasz Kazmierczak telemetry::getDbusReportPath(id), telemetry::reportInterface, 9639e6c388aSLukasz Kazmierczak [asyncResp, id = std::string(id), 964b14f357fSEd Tanous redfishMetrics = std::vector<nlohmann::json::object_t>(metrics.begin(), 9659e6c388aSLukasz Kazmierczak metrics.end())]( 9669e6c388aSLukasz Kazmierczak boost::system::error_code ec, 9679e6c388aSLukasz Kazmierczak const dbus::utility::DBusPropertiesMap& properties) mutable { 9689e6c388aSLukasz Kazmierczak if (!redfish::telemetry::verifyCommonErrors(asyncResp->res, id, ec)) 9699e6c388aSLukasz Kazmierczak { 9709e6c388aSLukasz Kazmierczak return; 9719e6c388aSLukasz Kazmierczak } 9729e6c388aSLukasz Kazmierczak 9739e6c388aSLukasz Kazmierczak ReadingParameters readingParams; 9749e6c388aSLukasz Kazmierczak 9759e6c388aSLukasz Kazmierczak const bool success = sdbusplus::unpackPropertiesNoThrow( 9769e6c388aSLukasz Kazmierczak dbus_utils::UnpackErrorPrinter(), properties, "ReadingParameters", 9779e6c388aSLukasz Kazmierczak readingParams); 9789e6c388aSLukasz Kazmierczak 9799e6c388aSLukasz Kazmierczak if (!success) 9809e6c388aSLukasz Kazmierczak { 9819e6c388aSLukasz Kazmierczak messages::internalError(asyncResp->res); 9829e6c388aSLukasz Kazmierczak return; 9839e6c388aSLukasz Kazmierczak } 9849e6c388aSLukasz Kazmierczak 9859e6c388aSLukasz Kazmierczak auto updateMetricsReq = std::make_shared<UpdateMetrics>(id, asyncResp); 9869e6c388aSLukasz Kazmierczak 9879e6c388aSLukasz Kazmierczak boost::container::flat_set<std::pair<std::string, std::string>> 9889e6c388aSLukasz Kazmierczak chassisSensors; 9899e6c388aSLukasz Kazmierczak 9909e6c388aSLukasz Kazmierczak size_t index = 0; 991b14f357fSEd Tanous for (nlohmann::json::object_t& metric : redfishMetrics) 9929e6c388aSLukasz Kazmierczak { 9939e6c388aSLukasz Kazmierczak AddReportArgs::MetricArgs metricArgs; 9949e6c388aSLukasz Kazmierczak std::vector< 9959e6c388aSLukasz Kazmierczak std::tuple<sdbusplus::message::object_path, std::string>> 9969e6c388aSLukasz Kazmierczak pathAndUri; 9979e6c388aSLukasz Kazmierczak 9989e6c388aSLukasz Kazmierczak if (index < readingParams.size()) 9999e6c388aSLukasz Kazmierczak { 10009e6c388aSLukasz Kazmierczak const ReadingParameters::value_type& existing = 10019e6c388aSLukasz Kazmierczak readingParams[index]; 10029e6c388aSLukasz Kazmierczak 10039e6c388aSLukasz Kazmierczak pathAndUri = std::get<0>(existing); 10049e6c388aSLukasz Kazmierczak metricArgs.collectionFunction = std::get<1>(existing); 10059e6c388aSLukasz Kazmierczak metricArgs.collectionTimeScope = std::get<2>(existing); 10069e6c388aSLukasz Kazmierczak metricArgs.collectionDuration = std::get<3>(existing); 10079e6c388aSLukasz Kazmierczak } 10089e6c388aSLukasz Kazmierczak 10099e6c388aSLukasz Kazmierczak if (!getUserMetric(asyncResp->res, metric, metricArgs)) 10109e6c388aSLukasz Kazmierczak { 10119e6c388aSLukasz Kazmierczak return; 10129e6c388aSLukasz Kazmierczak } 10139e6c388aSLukasz Kazmierczak 10149e6c388aSLukasz Kazmierczak std::optional<IncorrectMetricUri> error = 10159e6c388aSLukasz Kazmierczak getChassisSensorNode(metricArgs.uris, chassisSensors); 10169e6c388aSLukasz Kazmierczak 10179e6c388aSLukasz Kazmierczak if (error) 10189e6c388aSLukasz Kazmierczak { 10199e6c388aSLukasz Kazmierczak messages::propertyValueIncorrect( 10209e6c388aSLukasz Kazmierczak asyncResp->res, error->uri, 10219e6c388aSLukasz Kazmierczak "MetricProperties/" + std::to_string(error->index)); 10229e6c388aSLukasz Kazmierczak return; 10239e6c388aSLukasz Kazmierczak } 10249e6c388aSLukasz Kazmierczak 10259e6c388aSLukasz Kazmierczak updateMetricsReq->emplace(pathAndUri, metricArgs); 10269e6c388aSLukasz Kazmierczak index++; 10279e6c388aSLukasz Kazmierczak } 10289e6c388aSLukasz Kazmierczak 10299e6c388aSLukasz Kazmierczak for (const auto& [chassis, sensorType] : chassisSensors) 10309e6c388aSLukasz Kazmierczak { 10319e6c388aSLukasz Kazmierczak retrieveUriToDbusMap( 10329e6c388aSLukasz Kazmierczak chassis, sensorType, 10339e6c388aSLukasz Kazmierczak [asyncResp, updateMetricsReq]( 10349e6c388aSLukasz Kazmierczak const boost::beast::http::status status, 10359e6c388aSLukasz Kazmierczak const std::map<std::string, std::string>& uriToDbus) { 10369e6c388aSLukasz Kazmierczak if (status != boost::beast::http::status::ok) 10379e6c388aSLukasz Kazmierczak { 10389e6c388aSLukasz Kazmierczak BMCWEB_LOG_ERROR( 10399e6c388aSLukasz Kazmierczak "Failed to retrieve URI to dbus sensors map with err {}", 10409e6c388aSLukasz Kazmierczak static_cast<unsigned>(status)); 10419e6c388aSLukasz Kazmierczak return; 10429e6c388aSLukasz Kazmierczak } 10439e6c388aSLukasz Kazmierczak updateMetricsReq->insert(uriToDbus); 10449e6c388aSLukasz Kazmierczak }); 10459e6c388aSLukasz Kazmierczak } 10469e6c388aSLukasz Kazmierczak }); 10479e6c388aSLukasz Kazmierczak } 1048081ebf06SWludzik, Jozef 10494220be3bSEd Tanous inline void handleMetricReportDefinitionCollectionHead( 10504220be3bSEd Tanous App& app, const crow::Request& req, 10514220be3bSEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 10524220be3bSEd Tanous { 10534220be3bSEd Tanous if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 10544220be3bSEd Tanous { 10554220be3bSEd Tanous return; 10564220be3bSEd Tanous } 10574220be3bSEd Tanous asyncResp->res.addHeader( 10584220be3bSEd Tanous boost::beast::http::field::link, 10594220be3bSEd Tanous "</redfish/v1/JsonSchemas/MetricReportDefinitionCollection/MetricReportDefinitionCollection.json>; rel=describedby"); 10604220be3bSEd Tanous } 10614220be3bSEd Tanous 10624220be3bSEd Tanous inline void handleMetricReportDefinitionCollectionGet( 1063fc0edbe3SEd Tanous App& app, const crow::Request& req, 1064fc0edbe3SEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 1065fc0edbe3SEd Tanous { 1066fc0edbe3SEd Tanous if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 1067fc0edbe3SEd Tanous { 1068fc0edbe3SEd Tanous return; 1069fc0edbe3SEd Tanous } 10709e6c388aSLukasz Kazmierczak asyncResp->res.addHeader( 10719e6c388aSLukasz Kazmierczak boost::beast::http::field::link, 10729e6c388aSLukasz Kazmierczak "</redfish/v1/JsonSchemas/MetricReportDefinition/MetricReportDefinition.json>; rel=describedby"); 1073fc0edbe3SEd Tanous 1074fc0edbe3SEd Tanous asyncResp->res.jsonValue["@odata.type"] = 1075fc0edbe3SEd Tanous "#MetricReportDefinitionCollection." 1076fc0edbe3SEd Tanous "MetricReportDefinitionCollection"; 1077fc0edbe3SEd Tanous asyncResp->res.jsonValue["@odata.id"] = 1078fc0edbe3SEd Tanous "/redfish/v1/TelemetryService/MetricReportDefinitions"; 1079fc0edbe3SEd Tanous asyncResp->res.jsonValue["Name"] = "Metric Definition Collection"; 1080fc0edbe3SEd Tanous constexpr std::array<std::string_view, 1> interfaces{ 1081fc0edbe3SEd Tanous telemetry::reportInterface}; 1082fc0edbe3SEd Tanous collection_util::getCollectionMembers( 1083fc0edbe3SEd Tanous asyncResp, 1084fc0edbe3SEd Tanous boost::urls::url( 1085fc0edbe3SEd Tanous "/redfish/v1/TelemetryService/MetricReportDefinitions"), 1086fc0edbe3SEd Tanous interfaces, "/xyz/openbmc_project/Telemetry/Reports/TelemetryService"); 1087fc0edbe3SEd Tanous } 1088fc0edbe3SEd Tanous 108986a5ac98SEd Tanous inline void 10909e6c388aSLukasz Kazmierczak handleReportPatch(App& app, const crow::Request& req, 10919e6c388aSLukasz Kazmierczak const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 10929e6c388aSLukasz Kazmierczak std::string_view id) 10939e6c388aSLukasz Kazmierczak { 10949e6c388aSLukasz Kazmierczak if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 10959e6c388aSLukasz Kazmierczak { 10969e6c388aSLukasz Kazmierczak return; 10979e6c388aSLukasz Kazmierczak } 10989e6c388aSLukasz Kazmierczak 10999e6c388aSLukasz Kazmierczak std::optional<std::string> reportingTypeStr; 11009e6c388aSLukasz Kazmierczak std::optional<std::string> reportUpdatesStr; 11019e6c388aSLukasz Kazmierczak std::optional<bool> metricReportDefinitionEnabled; 1102b14f357fSEd Tanous std::optional<std::vector<nlohmann::json::object_t>> metrics; 11039e6c388aSLukasz Kazmierczak std::optional<std::vector<std::string>> reportActionsStr; 1104b14f357fSEd Tanous std::optional<std::string> scheduleDurationStr; 11059e6c388aSLukasz Kazmierczak 11069e6c388aSLukasz Kazmierczak if (!json_util::readJsonPatch( 11079e6c388aSLukasz Kazmierczak req, asyncResp->res, "Metrics", metrics, 11089e6c388aSLukasz Kazmierczak "MetricReportDefinitionType", reportingTypeStr, "ReportUpdates", 1109b14f357fSEd Tanous reportUpdatesStr, "ReportActions", reportActionsStr, 1110b14f357fSEd Tanous "Schedule/RecurrenceInterval", scheduleDurationStr, 1111b14f357fSEd Tanous "MetricReportDefinitionEnabled", metricReportDefinitionEnabled)) 11129e6c388aSLukasz Kazmierczak { 11139e6c388aSLukasz Kazmierczak return; 11149e6c388aSLukasz Kazmierczak } 11159e6c388aSLukasz Kazmierczak 11169e6c388aSLukasz Kazmierczak if (metricReportDefinitionEnabled) 11179e6c388aSLukasz Kazmierczak { 11189e6c388aSLukasz Kazmierczak setReportEnabled(asyncResp, id, *metricReportDefinitionEnabled); 11199e6c388aSLukasz Kazmierczak } 11209e6c388aSLukasz Kazmierczak 11219e6c388aSLukasz Kazmierczak if (reportUpdatesStr) 11229e6c388aSLukasz Kazmierczak { 11239e6c388aSLukasz Kazmierczak std::string dbusReportUpdates = toDbusReportUpdates(*reportUpdatesStr); 11249e6c388aSLukasz Kazmierczak if (dbusReportUpdates.empty()) 11259e6c388aSLukasz Kazmierczak { 11269e6c388aSLukasz Kazmierczak messages::propertyValueNotInList(asyncResp->res, *reportUpdatesStr, 11279e6c388aSLukasz Kazmierczak "ReportUpdates"); 11289e6c388aSLukasz Kazmierczak return; 11299e6c388aSLukasz Kazmierczak } 11309e6c388aSLukasz Kazmierczak setReportUpdates(asyncResp, id, dbusReportUpdates); 11319e6c388aSLukasz Kazmierczak } 11329e6c388aSLukasz Kazmierczak 11339e6c388aSLukasz Kazmierczak if (reportActionsStr) 11349e6c388aSLukasz Kazmierczak { 11359e6c388aSLukasz Kazmierczak std::vector<std::string> dbusReportActions; 11369e6c388aSLukasz Kazmierczak if (!toDbusReportActions(asyncResp->res, *reportActionsStr, 11379e6c388aSLukasz Kazmierczak dbusReportActions)) 11389e6c388aSLukasz Kazmierczak { 11399e6c388aSLukasz Kazmierczak return; 11409e6c388aSLukasz Kazmierczak } 11419e6c388aSLukasz Kazmierczak setReportActions(asyncResp, id, dbusReportActions); 11429e6c388aSLukasz Kazmierczak } 11439e6c388aSLukasz Kazmierczak 1144b14f357fSEd Tanous if (reportingTypeStr || scheduleDurationStr) 11459e6c388aSLukasz Kazmierczak { 11469e6c388aSLukasz Kazmierczak std::string dbusReportingType; 11479e6c388aSLukasz Kazmierczak if (reportingTypeStr) 11489e6c388aSLukasz Kazmierczak { 11499e6c388aSLukasz Kazmierczak dbusReportingType = toDbusReportingType(*reportingTypeStr); 11509e6c388aSLukasz Kazmierczak if (dbusReportingType.empty()) 11519e6c388aSLukasz Kazmierczak { 11529e6c388aSLukasz Kazmierczak messages::propertyValueNotInList(asyncResp->res, 11539e6c388aSLukasz Kazmierczak *reportingTypeStr, 11549e6c388aSLukasz Kazmierczak "MetricReportDefinitionType"); 11559e6c388aSLukasz Kazmierczak return; 11569e6c388aSLukasz Kazmierczak } 11579e6c388aSLukasz Kazmierczak } 11589e6c388aSLukasz Kazmierczak 11599e6c388aSLukasz Kazmierczak uint64_t recurrenceInterval = std::numeric_limits<uint64_t>::max(); 1160b14f357fSEd Tanous if (scheduleDurationStr) 11619e6c388aSLukasz Kazmierczak { 11629e6c388aSLukasz Kazmierczak std::optional<std::chrono::milliseconds> durationNum = 1163b14f357fSEd Tanous time_utils::fromDurationString(*scheduleDurationStr); 11649e6c388aSLukasz Kazmierczak if (!durationNum || durationNum->count() < 0) 11659e6c388aSLukasz Kazmierczak { 11669e6c388aSLukasz Kazmierczak messages::propertyValueIncorrect( 1167b14f357fSEd Tanous asyncResp->res, "RecurrenceInterval", *scheduleDurationStr); 11689e6c388aSLukasz Kazmierczak return; 11699e6c388aSLukasz Kazmierczak } 11709e6c388aSLukasz Kazmierczak 11719e6c388aSLukasz Kazmierczak recurrenceInterval = static_cast<uint64_t>(durationNum->count()); 11729e6c388aSLukasz Kazmierczak } 11739e6c388aSLukasz Kazmierczak 11749e6c388aSLukasz Kazmierczak setReportTypeAndInterval(asyncResp, id, dbusReportingType, 11759e6c388aSLukasz Kazmierczak recurrenceInterval); 11769e6c388aSLukasz Kazmierczak } 11779e6c388aSLukasz Kazmierczak 11789e6c388aSLukasz Kazmierczak if (metrics) 11799e6c388aSLukasz Kazmierczak { 11809e6c388aSLukasz Kazmierczak setReportMetrics(asyncResp, id, *metrics); 11819e6c388aSLukasz Kazmierczak } 11829e6c388aSLukasz Kazmierczak } 11839e6c388aSLukasz Kazmierczak 11849e6c388aSLukasz Kazmierczak inline void 11859e6c388aSLukasz Kazmierczak handleReportDelete(App& app, const crow::Request& req, 11869e6c388aSLukasz Kazmierczak const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 11879e6c388aSLukasz Kazmierczak std::string_view id) 11889e6c388aSLukasz Kazmierczak { 11899e6c388aSLukasz Kazmierczak if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 11909e6c388aSLukasz Kazmierczak { 11919e6c388aSLukasz Kazmierczak return; 11929e6c388aSLukasz Kazmierczak } 11939e6c388aSLukasz Kazmierczak 11949e6c388aSLukasz Kazmierczak const std::string reportPath = getDbusReportPath(id); 11959e6c388aSLukasz Kazmierczak 11969e6c388aSLukasz Kazmierczak crow::connections::systemBus->async_method_call( 11979e6c388aSLukasz Kazmierczak [asyncResp, 11989e6c388aSLukasz Kazmierczak reportId = std::string(id)](const boost::system::error_code& ec) { 11999e6c388aSLukasz Kazmierczak if (!verifyCommonErrors(asyncResp->res, reportId, ec)) 12009e6c388aSLukasz Kazmierczak { 12019e6c388aSLukasz Kazmierczak return; 12029e6c388aSLukasz Kazmierczak } 12039e6c388aSLukasz Kazmierczak asyncResp->res.result(boost::beast::http::status::no_content); 12049e6c388aSLukasz Kazmierczak }, 12059e6c388aSLukasz Kazmierczak service, reportPath, "xyz.openbmc_project.Object.Delete", "Delete"); 12069e6c388aSLukasz Kazmierczak } 12079e6c388aSLukasz Kazmierczak } // namespace telemetry 12089e6c388aSLukasz Kazmierczak 120995bdb5f0SEd Tanous inline void afterRetrieveUriToDbusMap( 121095bdb5f0SEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& /*asyncResp*/, 121195bdb5f0SEd Tanous const std::shared_ptr<telemetry::AddReport>& addReportReq, 121295bdb5f0SEd Tanous const boost::beast::http::status status, 121395bdb5f0SEd Tanous const std::map<std::string, std::string>& uriToDbus) 121495bdb5f0SEd Tanous { 121595bdb5f0SEd Tanous if (status != boost::beast::http::status::ok) 121695bdb5f0SEd Tanous { 121795bdb5f0SEd Tanous BMCWEB_LOG_ERROR( 121895bdb5f0SEd Tanous "Failed to retrieve URI to dbus sensors map with err {}", 121995bdb5f0SEd Tanous static_cast<unsigned>(status)); 122095bdb5f0SEd Tanous return; 122195bdb5f0SEd Tanous } 122295bdb5f0SEd Tanous addReportReq->insert(uriToDbus); 122395bdb5f0SEd Tanous } 122495bdb5f0SEd Tanous 12259e6c388aSLukasz Kazmierczak inline void handleMetricReportDefinitionsPost( 12269e6c388aSLukasz Kazmierczak App& app, const crow::Request& req, 12279e6c388aSLukasz Kazmierczak const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 12289e6c388aSLukasz Kazmierczak { 12299e6c388aSLukasz Kazmierczak if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 12309e6c388aSLukasz Kazmierczak { 12319e6c388aSLukasz Kazmierczak return; 12329e6c388aSLukasz Kazmierczak } 12339e6c388aSLukasz Kazmierczak 12349e6c388aSLukasz Kazmierczak telemetry::AddReportArgs args; 12359e6c388aSLukasz Kazmierczak if (!telemetry::getUserParameters(asyncResp->res, req, args)) 12369e6c388aSLukasz Kazmierczak { 12379e6c388aSLukasz Kazmierczak return; 12389e6c388aSLukasz Kazmierczak } 12399e6c388aSLukasz Kazmierczak 12409e6c388aSLukasz Kazmierczak boost::container::flat_set<std::pair<std::string, std::string>> 12419e6c388aSLukasz Kazmierczak chassisSensors; 12429e6c388aSLukasz Kazmierczak if (!telemetry::getChassisSensorNodeFromMetrics(asyncResp, args.metrics, 12439e6c388aSLukasz Kazmierczak chassisSensors)) 12449e6c388aSLukasz Kazmierczak { 12459e6c388aSLukasz Kazmierczak return; 12469e6c388aSLukasz Kazmierczak } 12479e6c388aSLukasz Kazmierczak 12489e6c388aSLukasz Kazmierczak auto addReportReq = std::make_shared<telemetry::AddReport>(std::move(args), 12499e6c388aSLukasz Kazmierczak asyncResp); 12509e6c388aSLukasz Kazmierczak for (const auto& [chassis, sensorType] : chassisSensors) 12519e6c388aSLukasz Kazmierczak { 125295bdb5f0SEd Tanous retrieveUriToDbusMap(chassis, sensorType, 125395bdb5f0SEd Tanous std::bind_front(afterRetrieveUriToDbusMap, 125495bdb5f0SEd Tanous asyncResp, addReportReq)); 12559e6c388aSLukasz Kazmierczak } 12569e6c388aSLukasz Kazmierczak } 12579e6c388aSLukasz Kazmierczak 12589e6c388aSLukasz Kazmierczak inline void 12594220be3bSEd Tanous handleMetricReportHead(App& app, const crow::Request& req, 12604220be3bSEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 12614220be3bSEd Tanous const std::string& /*id*/) 12624220be3bSEd Tanous { 12634220be3bSEd Tanous if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 12644220be3bSEd Tanous { 12654220be3bSEd Tanous return; 12664220be3bSEd Tanous } 12674220be3bSEd Tanous asyncResp->res.addHeader( 12684220be3bSEd Tanous boost::beast::http::field::link, 12694220be3bSEd Tanous "</redfish/v1/JsonSchemas/MetricReport/MetricReport.json>; rel=describedby"); 12704220be3bSEd Tanous } 12714220be3bSEd Tanous 12724220be3bSEd Tanous inline void 127386a5ac98SEd Tanous handleMetricReportGet(App& app, const crow::Request& req, 127486a5ac98SEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 127586a5ac98SEd Tanous const std::string& id) 127686a5ac98SEd Tanous { 127786a5ac98SEd Tanous if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 127886a5ac98SEd Tanous { 127986a5ac98SEd Tanous return; 128086a5ac98SEd Tanous } 12814220be3bSEd Tanous asyncResp->res.addHeader( 12824220be3bSEd Tanous boost::beast::http::field::link, 12834220be3bSEd Tanous "</redfish/v1/JsonSchemas/MetricReport/MetricReport.json>; rel=describedby"); 128486a5ac98SEd Tanous 128586a5ac98SEd Tanous sdbusplus::asio::getAllProperties( 128686a5ac98SEd Tanous *crow::connections::systemBus, telemetry::service, 128786a5ac98SEd Tanous telemetry::getDbusReportPath(id), telemetry::reportInterface, 128886a5ac98SEd Tanous [asyncResp, id](const boost::system::error_code& ec, 128986a5ac98SEd Tanous const dbus::utility::DBusPropertiesMap& properties) { 12909e6c388aSLukasz Kazmierczak if (!redfish::telemetry::verifyCommonErrors(asyncResp->res, id, ec)) 129186a5ac98SEd Tanous { 129286a5ac98SEd Tanous return; 129386a5ac98SEd Tanous } 129486a5ac98SEd Tanous 129586a5ac98SEd Tanous telemetry::fillReportDefinition(asyncResp, id, properties); 129686a5ac98SEd Tanous }); 129786a5ac98SEd Tanous } 129886a5ac98SEd Tanous 1299dd1c4a9cSSzymon Dompke inline void handleMetricReportDelete( 1300dd1c4a9cSSzymon Dompke App& app, const crow::Request& req, 1301dd1c4a9cSSzymon Dompke const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, const std::string& id) 1302dd1c4a9cSSzymon Dompke 1303dd1c4a9cSSzymon Dompke { 1304dd1c4a9cSSzymon Dompke if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 1305dd1c4a9cSSzymon Dompke { 1306dd1c4a9cSSzymon Dompke return; 1307dd1c4a9cSSzymon Dompke } 1308dd1c4a9cSSzymon Dompke 1309dd1c4a9cSSzymon Dompke const std::string reportPath = telemetry::getDbusReportPath(id); 1310dd1c4a9cSSzymon Dompke 1311dd1c4a9cSSzymon Dompke crow::connections::systemBus->async_method_call( 1312dd1c4a9cSSzymon Dompke [asyncResp, id](const boost::system::error_code& ec) { 1313dd1c4a9cSSzymon Dompke /* 1314dd1c4a9cSSzymon Dompke * boost::system::errc and std::errc are missing value 1315dd1c4a9cSSzymon Dompke * for EBADR error that is defined in Linux. 1316dd1c4a9cSSzymon Dompke */ 1317dd1c4a9cSSzymon Dompke if (ec.value() == EBADR) 1318dd1c4a9cSSzymon Dompke { 1319dd1c4a9cSSzymon Dompke messages::resourceNotFound(asyncResp->res, "MetricReportDefinition", 1320dd1c4a9cSSzymon Dompke id); 1321dd1c4a9cSSzymon Dompke return; 1322dd1c4a9cSSzymon Dompke } 1323dd1c4a9cSSzymon Dompke 1324dd1c4a9cSSzymon Dompke if (ec) 1325dd1c4a9cSSzymon Dompke { 132662598e31SEd Tanous BMCWEB_LOG_ERROR("respHandler DBus error {}", ec); 1327dd1c4a9cSSzymon Dompke messages::internalError(asyncResp->res); 1328dd1c4a9cSSzymon Dompke return; 1329dd1c4a9cSSzymon Dompke } 1330dd1c4a9cSSzymon Dompke 1331dd1c4a9cSSzymon Dompke asyncResp->res.result(boost::beast::http::status::no_content); 1332dd1c4a9cSSzymon Dompke }, 1333dd1c4a9cSSzymon Dompke telemetry::service, reportPath, "xyz.openbmc_project.Object.Delete", 1334dd1c4a9cSSzymon Dompke "Delete"); 1335dd1c4a9cSSzymon Dompke } 1336dd1c4a9cSSzymon Dompke 13377e860f15SJohn Edward Broadbent inline void requestRoutesMetricReportDefinitionCollection(App& app) 1338081ebf06SWludzik, Jozef { 13397e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions/") 13404220be3bSEd Tanous .privileges(redfish::privileges::headMetricReportDefinitionCollection) 13414220be3bSEd Tanous .methods(boost::beast::http::verb::head)(std::bind_front( 13429e6c388aSLukasz Kazmierczak telemetry::handleMetricReportDefinitionCollectionHead, 13439e6c388aSLukasz Kazmierczak std::ref(app))); 13444220be3bSEd Tanous 13454220be3bSEd Tanous BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions/") 1346ed398213SEd Tanous .privileges(redfish::privileges::getMetricReportDefinitionCollection) 13474220be3bSEd Tanous .methods(boost::beast::http::verb::get)(std::bind_front( 13489e6c388aSLukasz Kazmierczak telemetry::handleMetricReportDefinitionCollectionGet, 13499e6c388aSLukasz Kazmierczak std::ref(app))); 13504dbb8aeaSWludzik, Jozef 13517e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/TelemetryService/MetricReportDefinitions/") 1352ed398213SEd Tanous .privileges(redfish::privileges::postMetricReportDefinitionCollection) 1353002d39b4SEd Tanous .methods(boost::beast::http::verb::post)( 13549e6c388aSLukasz Kazmierczak std::bind_front(handleMetricReportDefinitionsPost, std::ref(app))); 1355081ebf06SWludzik, Jozef } 1356081ebf06SWludzik, Jozef 13577e860f15SJohn Edward Broadbent inline void requestRoutesMetricReportDefinition(App& app) 1358081ebf06SWludzik, Jozef { 13597e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, 13607e860f15SJohn Edward Broadbent "/redfish/v1/TelemetryService/MetricReportDefinitions/<str>/") 1361ed398213SEd Tanous .privileges(redfish::privileges::getMetricReportDefinition) 13624220be3bSEd Tanous .methods(boost::beast::http::verb::head)( 13634220be3bSEd Tanous std::bind_front(handleMetricReportHead, std::ref(app))); 13644220be3bSEd Tanous 13654220be3bSEd Tanous BMCWEB_ROUTE(app, 13664220be3bSEd Tanous "/redfish/v1/TelemetryService/MetricReportDefinitions/<str>/") 13674220be3bSEd Tanous .privileges(redfish::privileges::getMetricReportDefinition) 13687e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)( 136986a5ac98SEd Tanous std::bind_front(handleMetricReportGet, std::ref(app))); 1370479e899dSKrzysztof Grobelny 13717e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, 13727e860f15SJohn Edward Broadbent "/redfish/v1/TelemetryService/MetricReportDefinitions/<str>/") 13739e6c388aSLukasz Kazmierczak .privileges(redfish::privileges::deleteMetricReportDefinition) 13747e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::delete_)( 1375dd1c4a9cSSzymon Dompke std::bind_front(handleMetricReportDelete, std::ref(app))); 13769e6c388aSLukasz Kazmierczak 13779e6c388aSLukasz Kazmierczak BMCWEB_ROUTE(app, 13789e6c388aSLukasz Kazmierczak "/redfish/v1/TelemetryService/MetricReportDefinitions/<str>/") 13799e6c388aSLukasz Kazmierczak .privileges(redfish::privileges::patchMetricReportDefinition) 13809e6c388aSLukasz Kazmierczak .methods(boost::beast::http::verb::patch)( 13819e6c388aSLukasz Kazmierczak std::bind_front(telemetry::handleReportPatch, std::ref(app))); 13824dbb8aeaSWludzik, Jozef } 1383081ebf06SWludzik, Jozef } // namespace redfish 1384