19c310685SBorawski.Lukasz /* 29c310685SBorawski.Lukasz // Copyright (c) 2018 Intel Corporation 39c310685SBorawski.Lukasz // 49c310685SBorawski.Lukasz // Licensed under the Apache License, Version 2.0 (the "License"); 59c310685SBorawski.Lukasz // you may not use this file except in compliance with the License. 69c310685SBorawski.Lukasz // You may obtain a copy of the License at 79c310685SBorawski.Lukasz // 89c310685SBorawski.Lukasz // http://www.apache.org/licenses/LICENSE-2.0 99c310685SBorawski.Lukasz // 109c310685SBorawski.Lukasz // Unless required by applicable law or agreed to in writing, software 119c310685SBorawski.Lukasz // distributed under the License is distributed on an "AS IS" BASIS, 129c310685SBorawski.Lukasz // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 139c310685SBorawski.Lukasz // See the License for the specific language governing permissions and 149c310685SBorawski.Lukasz // limitations under the License. 159c310685SBorawski.Lukasz */ 169c310685SBorawski.Lukasz #pragma once 179c310685SBorawski.Lukasz 18a51fc2d2SSui Chen #include "app.hpp" 19a51fc2d2SSui Chen #include "dbus_utility.hpp" 20b49ac873SJames Feist #include "health.hpp" 21a51fc2d2SSui Chen #include "query.hpp" 22c5d03ff4SJennifer Lee #include "redfish_util.hpp" 23a51fc2d2SSui Chen #include "registries/privilege_registry.hpp" 24fac6e53bSKrzysztof Grobelny #include "utils/dbus_utils.hpp" 25*3ccb3adbSEd Tanous #include "utils/json_utils.hpp" 26a51fc2d2SSui Chen #include "utils/sw_utils.hpp" 27a51fc2d2SSui Chen #include "utils/systemd_utils.hpp" 282b82937eSEd Tanous #include "utils/time_utils.hpp" 299c310685SBorawski.Lukasz 30af5d6058SSantosh Puranik #include <boost/date_time.hpp> 31fac6e53bSKrzysztof Grobelny #include <sdbusplus/asio/property.hpp> 32fac6e53bSKrzysztof Grobelny #include <sdbusplus/unpack_properties.hpp> 331214b7e7SGunnar Mills 34a170f275SEd Tanous #include <algorithm> 354bfefa74SGunnar Mills #include <cstdint> 361214b7e7SGunnar Mills #include <memory> 371214b7e7SGunnar Mills #include <sstream> 38abf2add6SEd Tanous #include <variant> 395b4aa86bSJames Feist 401abe55efSEd Tanous namespace redfish 411abe55efSEd Tanous { 42ed5befbdSJennifer Lee 43ed5befbdSJennifer Lee /** 442a5c4407SGunnar Mills * Function reboots the BMC. 452a5c4407SGunnar Mills * 462a5c4407SGunnar Mills * @param[in] asyncResp - Shared pointer for completing asynchronous calls 47ed5befbdSJennifer Lee */ 488d1b46d7Szhanghch05 inline void 498d1b46d7Szhanghch05 doBMCGracefulRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 50ed5befbdSJennifer Lee { 51ed5befbdSJennifer Lee const char* processName = "xyz.openbmc_project.State.BMC"; 52ed5befbdSJennifer Lee const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 53ed5befbdSJennifer Lee const char* interfaceName = "xyz.openbmc_project.State.BMC"; 54ed5befbdSJennifer Lee const std::string& propertyValue = 55ed5befbdSJennifer Lee "xyz.openbmc_project.State.BMC.Transition.Reboot"; 56ed5befbdSJennifer Lee const char* destProperty = "RequestedBMCTransition"; 57ed5befbdSJennifer Lee 58ed5befbdSJennifer Lee // Create the D-Bus variant for D-Bus call. 59168e20c1SEd Tanous dbus::utility::DbusVariantType dbusPropertyValue(propertyValue); 60ed5befbdSJennifer Lee 61ed5befbdSJennifer Lee crow::connections::systemBus->async_method_call( 62ed5befbdSJennifer Lee [asyncResp](const boost::system::error_code ec) { 63ed5befbdSJennifer Lee // Use "Set" method to set the property value. 64ed5befbdSJennifer Lee if (ec) 65ed5befbdSJennifer Lee { 662a5c4407SGunnar Mills BMCWEB_LOG_DEBUG << "[Set] Bad D-Bus request error: " << ec; 67ed5befbdSJennifer Lee messages::internalError(asyncResp->res); 68ed5befbdSJennifer Lee return; 69ed5befbdSJennifer Lee } 70ed5befbdSJennifer Lee 71ed5befbdSJennifer Lee messages::success(asyncResp->res); 72ed5befbdSJennifer Lee }, 73ed5befbdSJennifer Lee processName, objectPath, "org.freedesktop.DBus.Properties", "Set", 74ed5befbdSJennifer Lee interfaceName, destProperty, dbusPropertyValue); 75ed5befbdSJennifer Lee } 762a5c4407SGunnar Mills 778d1b46d7Szhanghch05 inline void 788d1b46d7Szhanghch05 doBMCForceRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 79f92af389SJayaprakash Mutyala { 80f92af389SJayaprakash Mutyala const char* processName = "xyz.openbmc_project.State.BMC"; 81f92af389SJayaprakash Mutyala const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 82f92af389SJayaprakash Mutyala const char* interfaceName = "xyz.openbmc_project.State.BMC"; 83f92af389SJayaprakash Mutyala const std::string& propertyValue = 84f92af389SJayaprakash Mutyala "xyz.openbmc_project.State.BMC.Transition.HardReboot"; 85f92af389SJayaprakash Mutyala const char* destProperty = "RequestedBMCTransition"; 86f92af389SJayaprakash Mutyala 87f92af389SJayaprakash Mutyala // Create the D-Bus variant for D-Bus call. 88168e20c1SEd Tanous dbus::utility::DbusVariantType dbusPropertyValue(propertyValue); 89f92af389SJayaprakash Mutyala 90f92af389SJayaprakash Mutyala crow::connections::systemBus->async_method_call( 91f92af389SJayaprakash Mutyala [asyncResp](const boost::system::error_code ec) { 92f92af389SJayaprakash Mutyala // Use "Set" method to set the property value. 93f92af389SJayaprakash Mutyala if (ec) 94f92af389SJayaprakash Mutyala { 95f92af389SJayaprakash Mutyala BMCWEB_LOG_DEBUG << "[Set] Bad D-Bus request error: " << ec; 96f92af389SJayaprakash Mutyala messages::internalError(asyncResp->res); 97f92af389SJayaprakash Mutyala return; 98f92af389SJayaprakash Mutyala } 99f92af389SJayaprakash Mutyala 100f92af389SJayaprakash Mutyala messages::success(asyncResp->res); 101f92af389SJayaprakash Mutyala }, 102f92af389SJayaprakash Mutyala processName, objectPath, "org.freedesktop.DBus.Properties", "Set", 103f92af389SJayaprakash Mutyala interfaceName, destProperty, dbusPropertyValue); 104f92af389SJayaprakash Mutyala } 105f92af389SJayaprakash Mutyala 1062a5c4407SGunnar Mills /** 1072a5c4407SGunnar Mills * ManagerResetAction class supports the POST method for the Reset (reboot) 1082a5c4407SGunnar Mills * action. 1092a5c4407SGunnar Mills */ 1107e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetAction(App& app) 1112a5c4407SGunnar Mills { 1122a5c4407SGunnar Mills /** 1132a5c4407SGunnar Mills * Function handles POST method request. 1142a5c4407SGunnar Mills * Analyzes POST body before sending Reset (Reboot) request data to D-Bus. 115f92af389SJayaprakash Mutyala * OpenBMC supports ResetType "GracefulRestart" and "ForceRestart". 1162a5c4407SGunnar Mills */ 1177e860f15SJohn Edward Broadbent 1187e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/") 119ed398213SEd Tanous .privileges(redfish::privileges::postManager) 1207e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::post)( 12145ca1b86SEd Tanous [&app](const crow::Request& req, 1227e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 1233ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 12445ca1b86SEd Tanous { 12545ca1b86SEd Tanous return; 12645ca1b86SEd Tanous } 1272a5c4407SGunnar Mills BMCWEB_LOG_DEBUG << "Post Manager Reset."; 1282a5c4407SGunnar Mills 1292a5c4407SGunnar Mills std::string resetType; 1302a5c4407SGunnar Mills 13115ed6780SWilly Tu if (!json_util::readJsonAction(req, asyncResp->res, "ResetType", 1327e860f15SJohn Edward Broadbent resetType)) 1332a5c4407SGunnar Mills { 1342a5c4407SGunnar Mills return; 1352a5c4407SGunnar Mills } 1362a5c4407SGunnar Mills 137f92af389SJayaprakash Mutyala if (resetType == "GracefulRestart") 138f92af389SJayaprakash Mutyala { 139f92af389SJayaprakash Mutyala BMCWEB_LOG_DEBUG << "Proceeding with " << resetType; 140f92af389SJayaprakash Mutyala doBMCGracefulRestart(asyncResp); 141f92af389SJayaprakash Mutyala return; 142f92af389SJayaprakash Mutyala } 1433174e4dfSEd Tanous if (resetType == "ForceRestart") 144f92af389SJayaprakash Mutyala { 145f92af389SJayaprakash Mutyala BMCWEB_LOG_DEBUG << "Proceeding with " << resetType; 146f92af389SJayaprakash Mutyala doBMCForceRestart(asyncResp); 147f92af389SJayaprakash Mutyala return; 148f92af389SJayaprakash Mutyala } 1492a5c4407SGunnar Mills BMCWEB_LOG_DEBUG << "Invalid property value for ResetType: " 1502a5c4407SGunnar Mills << resetType; 1512a5c4407SGunnar Mills messages::actionParameterNotSupported(asyncResp->res, resetType, 1522a5c4407SGunnar Mills "ResetType"); 1532a5c4407SGunnar Mills 1542a5c4407SGunnar Mills return; 1557e860f15SJohn Edward Broadbent }); 1562a5c4407SGunnar Mills } 157ed5befbdSJennifer Lee 1583e40fc74SGunnar Mills /** 1593e40fc74SGunnar Mills * ManagerResetToDefaultsAction class supports POST method for factory reset 1603e40fc74SGunnar Mills * action. 1613e40fc74SGunnar Mills */ 1627e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetToDefaultsAction(App& app) 1633e40fc74SGunnar Mills { 1643e40fc74SGunnar Mills 1653e40fc74SGunnar Mills /** 1663e40fc74SGunnar Mills * Function handles ResetToDefaults POST method request. 1673e40fc74SGunnar Mills * 1683e40fc74SGunnar Mills * Analyzes POST body message and factory resets BMC by calling 1693e40fc74SGunnar Mills * BMC code updater factory reset followed by a BMC reboot. 1703e40fc74SGunnar Mills * 1713e40fc74SGunnar Mills * BMC code updater factory reset wipes the whole BMC read-write 1723e40fc74SGunnar Mills * filesystem which includes things like the network settings. 1733e40fc74SGunnar Mills * 1743e40fc74SGunnar Mills * OpenBMC only supports ResetToDefaultsType "ResetAll". 1753e40fc74SGunnar Mills */ 1767e860f15SJohn Edward Broadbent 1777e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, 1787e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults/") 179ed398213SEd Tanous .privileges(redfish::privileges::postManager) 1807e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::post)( 18145ca1b86SEd Tanous [&app](const crow::Request& req, 1827e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 1833ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 18445ca1b86SEd Tanous { 18545ca1b86SEd Tanous return; 18645ca1b86SEd Tanous } 1873e40fc74SGunnar Mills BMCWEB_LOG_DEBUG << "Post ResetToDefaults."; 1883e40fc74SGunnar Mills 1893e40fc74SGunnar Mills std::string resetType; 1903e40fc74SGunnar Mills 191002d39b4SEd Tanous if (!json_util::readJsonAction(req, asyncResp->res, 192002d39b4SEd Tanous "ResetToDefaultsType", resetType)) 1933e40fc74SGunnar Mills { 1943e40fc74SGunnar Mills BMCWEB_LOG_DEBUG << "Missing property ResetToDefaultsType."; 1953e40fc74SGunnar Mills 196002d39b4SEd Tanous messages::actionParameterMissing(asyncResp->res, "ResetToDefaults", 1973e40fc74SGunnar Mills "ResetToDefaultsType"); 1983e40fc74SGunnar Mills return; 1993e40fc74SGunnar Mills } 2003e40fc74SGunnar Mills 2013e40fc74SGunnar Mills if (resetType != "ResetAll") 2023e40fc74SGunnar Mills { 2030fda0f12SGeorge Liu BMCWEB_LOG_DEBUG 2040fda0f12SGeorge Liu << "Invalid property value for ResetToDefaultsType: " 2053e40fc74SGunnar Mills << resetType; 206002d39b4SEd Tanous messages::actionParameterNotSupported(asyncResp->res, resetType, 207002d39b4SEd Tanous "ResetToDefaultsType"); 2083e40fc74SGunnar Mills return; 2093e40fc74SGunnar Mills } 2103e40fc74SGunnar Mills 2113e40fc74SGunnar Mills crow::connections::systemBus->async_method_call( 2123e40fc74SGunnar Mills [asyncResp](const boost::system::error_code ec) { 2133e40fc74SGunnar Mills if (ec) 2143e40fc74SGunnar Mills { 215002d39b4SEd Tanous BMCWEB_LOG_DEBUG << "Failed to ResetToDefaults: " << ec; 2163e40fc74SGunnar Mills messages::internalError(asyncResp->res); 2173e40fc74SGunnar Mills return; 2183e40fc74SGunnar Mills } 2193e40fc74SGunnar Mills // Factory Reset doesn't actually happen until a reboot 2203e40fc74SGunnar Mills // Can't erase what the BMC is running on 2213e40fc74SGunnar Mills doBMCGracefulRestart(asyncResp); 2223e40fc74SGunnar Mills }, 2233e40fc74SGunnar Mills "xyz.openbmc_project.Software.BMC.Updater", 2243e40fc74SGunnar Mills "/xyz/openbmc_project/software", 2253e40fc74SGunnar Mills "xyz.openbmc_project.Common.FactoryReset", "Reset"); 2267e860f15SJohn Edward Broadbent }); 2273e40fc74SGunnar Mills } 2283e40fc74SGunnar Mills 2291cb1a9e6SAppaRao Puli /** 2301cb1a9e6SAppaRao Puli * ManagerResetActionInfo derived class for delivering Manager 2311cb1a9e6SAppaRao Puli * ResetType AllowableValues using ResetInfo schema. 2321cb1a9e6SAppaRao Puli */ 2337e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetActionInfo(App& app) 2341cb1a9e6SAppaRao Puli { 2351cb1a9e6SAppaRao Puli /** 2361cb1a9e6SAppaRao Puli * Functions triggers appropriate requests on DBus 2371cb1a9e6SAppaRao Puli */ 2387e860f15SJohn Edward Broadbent 2397e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/ResetActionInfo/") 240ed398213SEd Tanous .privileges(redfish::privileges::getActionInfo) 2417e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)( 24245ca1b86SEd Tanous [&app](const crow::Request& req, 2437e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 2443ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 24545ca1b86SEd Tanous { 24645ca1b86SEd Tanous return; 24745ca1b86SEd Tanous } 2481476687dSEd Tanous 2491476687dSEd Tanous asyncResp->res.jsonValue["@odata.type"] = 2501476687dSEd Tanous "#ActionInfo.v1_1_2.ActionInfo"; 2511476687dSEd Tanous asyncResp->res.jsonValue["@odata.id"] = 2521476687dSEd Tanous "/redfish/v1/Managers/bmc/ResetActionInfo"; 2531476687dSEd Tanous asyncResp->res.jsonValue["Name"] = "Reset Action Info"; 2541476687dSEd Tanous asyncResp->res.jsonValue["Id"] = "ResetActionInfo"; 2551476687dSEd Tanous nlohmann::json::object_t parameter; 2561476687dSEd Tanous parameter["Name"] = "ResetType"; 2571476687dSEd Tanous parameter["Required"] = true; 2581476687dSEd Tanous parameter["DataType"] = "String"; 2591476687dSEd Tanous 2601476687dSEd Tanous nlohmann::json::array_t allowableValues; 2611476687dSEd Tanous allowableValues.push_back("GracefulRestart"); 2621476687dSEd Tanous allowableValues.push_back("ForceRestart"); 2631476687dSEd Tanous parameter["AllowableValues"] = std::move(allowableValues); 2641476687dSEd Tanous 2651476687dSEd Tanous nlohmann::json::array_t parameters; 2661476687dSEd Tanous parameters.push_back(std::move(parameter)); 2671476687dSEd Tanous 2681476687dSEd Tanous asyncResp->res.jsonValue["Parameters"] = std::move(parameters); 2697e860f15SJohn Edward Broadbent }); 2701cb1a9e6SAppaRao Puli } 2711cb1a9e6SAppaRao Puli 2725b4aa86bSJames Feist static constexpr const char* objectManagerIface = 2735b4aa86bSJames Feist "org.freedesktop.DBus.ObjectManager"; 2745b4aa86bSJames Feist static constexpr const char* pidConfigurationIface = 2755b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid"; 2765b4aa86bSJames Feist static constexpr const char* pidZoneConfigurationIface = 2775b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid.Zone"; 278b7a08d04SJames Feist static constexpr const char* stepwiseConfigurationIface = 279b7a08d04SJames Feist "xyz.openbmc_project.Configuration.Stepwise"; 28073df0db0SJames Feist static constexpr const char* thermalModeIface = 28173df0db0SJames Feist "xyz.openbmc_project.Control.ThermalMode"; 2829c310685SBorawski.Lukasz 2838d1b46d7Szhanghch05 inline void 2848d1b46d7Szhanghch05 asyncPopulatePid(const std::string& connection, const std::string& path, 28573df0db0SJames Feist const std::string& currentProfile, 28673df0db0SJames Feist const std::vector<std::string>& supportedProfiles, 2878d1b46d7Szhanghch05 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 2885b4aa86bSJames Feist { 2895b4aa86bSJames Feist 2905b4aa86bSJames Feist crow::connections::systemBus->async_method_call( 29173df0db0SJames Feist [asyncResp, currentProfile, supportedProfiles]( 29273df0db0SJames Feist const boost::system::error_code ec, 2935b4aa86bSJames Feist const dbus::utility::ManagedObjectType& managedObj) { 2945b4aa86bSJames Feist if (ec) 2955b4aa86bSJames Feist { 2965b4aa86bSJames Feist BMCWEB_LOG_ERROR << ec; 2975b4aa86bSJames Feist asyncResp->res.jsonValue.clear(); 298f12894f8SJason M. Bills messages::internalError(asyncResp->res); 2995b4aa86bSJames Feist return; 3005b4aa86bSJames Feist } 3015b4aa86bSJames Feist nlohmann::json& configRoot = 3025b4aa86bSJames Feist asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"]; 3035b4aa86bSJames Feist nlohmann::json& fans = configRoot["FanControllers"]; 3045b4aa86bSJames Feist fans["@odata.type"] = "#OemManager.FanControllers"; 3050fda0f12SGeorge Liu fans["@odata.id"] = 3060fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanControllers"; 3075b4aa86bSJames Feist 3085b4aa86bSJames Feist nlohmann::json& pids = configRoot["PidControllers"]; 3095b4aa86bSJames Feist pids["@odata.type"] = "#OemManager.PidControllers"; 3105b4aa86bSJames Feist pids["@odata.id"] = 3115b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers"; 3125b4aa86bSJames Feist 313b7a08d04SJames Feist nlohmann::json& stepwise = configRoot["StepwiseControllers"]; 314b7a08d04SJames Feist stepwise["@odata.type"] = "#OemManager.StepwiseControllers"; 315b7a08d04SJames Feist stepwise["@odata.id"] = 316b7a08d04SJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers"; 317b7a08d04SJames Feist 3185b4aa86bSJames Feist nlohmann::json& zones = configRoot["FanZones"]; 3195b4aa86bSJames Feist zones["@odata.id"] = 3205b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones"; 3215b4aa86bSJames Feist zones["@odata.type"] = "#OemManager.FanZones"; 322002d39b4SEd Tanous configRoot["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan"; 3235b4aa86bSJames Feist configRoot["@odata.type"] = "#OemManager.Fan"; 32473df0db0SJames Feist configRoot["Profile@Redfish.AllowableValues"] = supportedProfiles; 32573df0db0SJames Feist 32673df0db0SJames Feist if (!currentProfile.empty()) 32773df0db0SJames Feist { 32873df0db0SJames Feist configRoot["Profile"] = currentProfile; 32973df0db0SJames Feist } 33073df0db0SJames Feist BMCWEB_LOG_ERROR << "profile = " << currentProfile << " !"; 3315b4aa86bSJames Feist 3325b4aa86bSJames Feist for (const auto& pathPair : managedObj) 3335b4aa86bSJames Feist { 3345b4aa86bSJames Feist for (const auto& intfPair : pathPair.second) 3355b4aa86bSJames Feist { 3365b4aa86bSJames Feist if (intfPair.first != pidConfigurationIface && 337b7a08d04SJames Feist intfPair.first != pidZoneConfigurationIface && 338b7a08d04SJames Feist intfPair.first != stepwiseConfigurationIface) 3395b4aa86bSJames Feist { 3405b4aa86bSJames Feist continue; 3415b4aa86bSJames Feist } 34273df0db0SJames Feist 343711ac7a9SEd Tanous std::string name; 344711ac7a9SEd Tanous 345711ac7a9SEd Tanous for (const std::pair<std::string, 346002d39b4SEd Tanous dbus::utility::DbusVariantType>& propPair : 347002d39b4SEd Tanous intfPair.second) 348711ac7a9SEd Tanous { 349711ac7a9SEd Tanous if (propPair.first == "Name") 350711ac7a9SEd Tanous { 3515b4aa86bSJames Feist const std::string* namePtr = 352711ac7a9SEd Tanous std::get_if<std::string>(&propPair.second); 3535b4aa86bSJames Feist if (namePtr == nullptr) 3545b4aa86bSJames Feist { 3555b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Name Field illegal"; 356b7a08d04SJames Feist messages::internalError(asyncResp->res); 3575b4aa86bSJames Feist return; 3585b4aa86bSJames Feist } 359db697703SWilly Tu name = *namePtr; 3605b4aa86bSJames Feist dbus::utility::escapePathForDbus(name); 361711ac7a9SEd Tanous } 362711ac7a9SEd Tanous else if (propPair.first == "Profiles") 36373df0db0SJames Feist { 36473df0db0SJames Feist const std::vector<std::string>* profiles = 36573df0db0SJames Feist std::get_if<std::vector<std::string>>( 366711ac7a9SEd Tanous &propPair.second); 36773df0db0SJames Feist if (profiles == nullptr) 36873df0db0SJames Feist { 369002d39b4SEd Tanous BMCWEB_LOG_ERROR << "Pid Profiles Field illegal"; 37073df0db0SJames Feist messages::internalError(asyncResp->res); 37173df0db0SJames Feist return; 37273df0db0SJames Feist } 37373df0db0SJames Feist if (std::find(profiles->begin(), profiles->end(), 37473df0db0SJames Feist currentProfile) == profiles->end()) 37573df0db0SJames Feist { 37673df0db0SJames Feist BMCWEB_LOG_INFO 377002d39b4SEd Tanous << name << " not supported in current profile"; 37873df0db0SJames Feist continue; 37973df0db0SJames Feist } 38073df0db0SJames Feist } 381711ac7a9SEd Tanous } 382b7a08d04SJames Feist nlohmann::json* config = nullptr; 383c33a90ecSJames Feist const std::string* classPtr = nullptr; 384711ac7a9SEd Tanous 385711ac7a9SEd Tanous for (const std::pair<std::string, 386002d39b4SEd Tanous dbus::utility::DbusVariantType>& propPair : 387002d39b4SEd Tanous intfPair.second) 388c33a90ecSJames Feist { 389727dc83fSLei YU if (propPair.first == "Class") 390711ac7a9SEd Tanous { 391002d39b4SEd Tanous classPtr = std::get_if<std::string>(&propPair.second); 392711ac7a9SEd Tanous } 393c33a90ecSJames Feist } 394c33a90ecSJames Feist 3955b4aa86bSJames Feist if (intfPair.first == pidZoneConfigurationIface) 3965b4aa86bSJames Feist { 3975b4aa86bSJames Feist std::string chassis; 398002d39b4SEd Tanous if (!dbus::utility::getNthStringFromPath(pathPair.first.str, 399002d39b4SEd Tanous 5, chassis)) 4005b4aa86bSJames Feist { 4015b4aa86bSJames Feist chassis = "#IllegalValue"; 4025b4aa86bSJames Feist } 4035b4aa86bSJames Feist nlohmann::json& zone = zones[name]; 404613dabeaSEd Tanous zone["Chassis"]["@odata.id"] = 405613dabeaSEd Tanous "/redfish/v1/Chassis/" + chassis; 4060fda0f12SGeorge Liu zone["@odata.id"] = 4070fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/" + 4085b4aa86bSJames Feist name; 4095b4aa86bSJames Feist zone["@odata.type"] = "#OemManager.FanZone"; 410b7a08d04SJames Feist config = &zone; 4115b4aa86bSJames Feist } 4125b4aa86bSJames Feist 413b7a08d04SJames Feist else if (intfPair.first == stepwiseConfigurationIface) 4145b4aa86bSJames Feist { 415c33a90ecSJames Feist if (classPtr == nullptr) 416c33a90ecSJames Feist { 417c33a90ecSJames Feist BMCWEB_LOG_ERROR << "Pid Class Field illegal"; 418c33a90ecSJames Feist messages::internalError(asyncResp->res); 419c33a90ecSJames Feist return; 420c33a90ecSJames Feist } 421c33a90ecSJames Feist 422b7a08d04SJames Feist nlohmann::json& controller = stepwise[name]; 423b7a08d04SJames Feist config = &controller; 4245b4aa86bSJames Feist 425b7a08d04SJames Feist controller["@odata.id"] = 4260fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers/" + 427271584abSEd Tanous name; 428b7a08d04SJames Feist controller["@odata.type"] = 429b7a08d04SJames Feist "#OemManager.StepwiseController"; 430b7a08d04SJames Feist 431c33a90ecSJames Feist controller["Direction"] = *classPtr; 4325b4aa86bSJames Feist } 4335b4aa86bSJames Feist 4345b4aa86bSJames Feist // pid and fans are off the same configuration 435b7a08d04SJames Feist else if (intfPair.first == pidConfigurationIface) 4365b4aa86bSJames Feist { 437c33a90ecSJames Feist 4385b4aa86bSJames Feist if (classPtr == nullptr) 4395b4aa86bSJames Feist { 4405b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Class Field illegal"; 441a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 4425b4aa86bSJames Feist return; 4435b4aa86bSJames Feist } 4445b4aa86bSJames Feist bool isFan = *classPtr == "fan"; 445002d39b4SEd Tanous nlohmann::json& element = isFan ? fans[name] : pids[name]; 446b7a08d04SJames Feist config = &element; 4475b4aa86bSJames Feist if (isFan) 4485b4aa86bSJames Feist { 4495b4aa86bSJames Feist element["@odata.id"] = 4500fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanControllers/" + 451271584abSEd Tanous name; 452002d39b4SEd Tanous element["@odata.type"] = "#OemManager.FanController"; 4535b4aa86bSJames Feist } 4545b4aa86bSJames Feist else 4555b4aa86bSJames Feist { 4565b4aa86bSJames Feist element["@odata.id"] = 4570fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers/" + 458271584abSEd Tanous name; 459002d39b4SEd Tanous element["@odata.type"] = "#OemManager.PidController"; 4605b4aa86bSJames Feist } 461b7a08d04SJames Feist } 462b7a08d04SJames Feist else 463b7a08d04SJames Feist { 464b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Unexpected configuration"; 465b7a08d04SJames Feist messages::internalError(asyncResp->res); 466b7a08d04SJames Feist return; 467b7a08d04SJames Feist } 468b7a08d04SJames Feist 469b7a08d04SJames Feist // used for making maps out of 2 vectors 470b7a08d04SJames Feist const std::vector<double>* keys = nullptr; 471b7a08d04SJames Feist const std::vector<double>* values = nullptr; 472b7a08d04SJames Feist 473b7a08d04SJames Feist for (const auto& propertyPair : intfPair.second) 474b7a08d04SJames Feist { 475b7a08d04SJames Feist if (propertyPair.first == "Type" || 476b7a08d04SJames Feist propertyPair.first == "Class" || 477b7a08d04SJames Feist propertyPair.first == "Name") 478b7a08d04SJames Feist { 479b7a08d04SJames Feist continue; 480b7a08d04SJames Feist } 481b7a08d04SJames Feist 482b7a08d04SJames Feist // zones 483b7a08d04SJames Feist if (intfPair.first == pidZoneConfigurationIface) 484b7a08d04SJames Feist { 485b7a08d04SJames Feist const double* ptr = 486abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 487b7a08d04SJames Feist if (ptr == nullptr) 488b7a08d04SJames Feist { 489b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 490b7a08d04SJames Feist << propertyPair.first; 491b7a08d04SJames Feist messages::internalError(asyncResp->res); 492b7a08d04SJames Feist return; 493b7a08d04SJames Feist } 494b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 495b7a08d04SJames Feist } 496b7a08d04SJames Feist 497b7a08d04SJames Feist if (intfPair.first == stepwiseConfigurationIface) 498b7a08d04SJames Feist { 499b7a08d04SJames Feist if (propertyPair.first == "Reading" || 500b7a08d04SJames Feist propertyPair.first == "Output") 501b7a08d04SJames Feist { 502b7a08d04SJames Feist const std::vector<double>* ptr = 503abf2add6SEd Tanous std::get_if<std::vector<double>>( 504b7a08d04SJames Feist &propertyPair.second); 505b7a08d04SJames Feist 506b7a08d04SJames Feist if (ptr == nullptr) 507b7a08d04SJames Feist { 508b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 509b7a08d04SJames Feist << propertyPair.first; 510b7a08d04SJames Feist messages::internalError(asyncResp->res); 511b7a08d04SJames Feist return; 512b7a08d04SJames Feist } 513b7a08d04SJames Feist 514b7a08d04SJames Feist if (propertyPair.first == "Reading") 515b7a08d04SJames Feist { 516b7a08d04SJames Feist keys = ptr; 517b7a08d04SJames Feist } 518b7a08d04SJames Feist else 519b7a08d04SJames Feist { 520b7a08d04SJames Feist values = ptr; 521b7a08d04SJames Feist } 522e662eae8SEd Tanous if (keys != nullptr && values != nullptr) 523b7a08d04SJames Feist { 524b7a08d04SJames Feist if (keys->size() != values->size()) 525b7a08d04SJames Feist { 526b7a08d04SJames Feist BMCWEB_LOG_ERROR 5270fda0f12SGeorge Liu << "Reading and Output size don't match "; 528b7a08d04SJames Feist messages::internalError(asyncResp->res); 529b7a08d04SJames Feist return; 530b7a08d04SJames Feist } 531b7a08d04SJames Feist nlohmann::json& steps = (*config)["Steps"]; 532b7a08d04SJames Feist steps = nlohmann::json::array(); 533b7a08d04SJames Feist for (size_t ii = 0; ii < keys->size(); ii++) 534b7a08d04SJames Feist { 5351476687dSEd Tanous nlohmann::json::object_t step; 5361476687dSEd Tanous step["Target"] = (*keys)[ii]; 5371476687dSEd Tanous step["Output"] = (*values)[ii]; 5381476687dSEd Tanous steps.push_back(std::move(step)); 539b7a08d04SJames Feist } 540b7a08d04SJames Feist } 541b7a08d04SJames Feist } 542b7a08d04SJames Feist if (propertyPair.first == "NegativeHysteresis" || 543b7a08d04SJames Feist propertyPair.first == "PositiveHysteresis") 544b7a08d04SJames Feist { 545b7a08d04SJames Feist const double* ptr = 546abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 547b7a08d04SJames Feist if (ptr == nullptr) 548b7a08d04SJames Feist { 549b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 550b7a08d04SJames Feist << propertyPair.first; 551b7a08d04SJames Feist messages::internalError(asyncResp->res); 552b7a08d04SJames Feist return; 553b7a08d04SJames Feist } 554b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 555b7a08d04SJames Feist } 556b7a08d04SJames Feist } 557b7a08d04SJames Feist 558b7a08d04SJames Feist // pid and fans are off the same configuration 559b7a08d04SJames Feist if (intfPair.first == pidConfigurationIface || 560b7a08d04SJames Feist intfPair.first == stepwiseConfigurationIface) 561b7a08d04SJames Feist { 5625b4aa86bSJames Feist 5635b4aa86bSJames Feist if (propertyPair.first == "Zones") 5645b4aa86bSJames Feist { 5655b4aa86bSJames Feist const std::vector<std::string>* inputs = 566abf2add6SEd Tanous std::get_if<std::vector<std::string>>( 5671b6b96c5SEd Tanous &propertyPair.second); 5685b4aa86bSJames Feist 5695b4aa86bSJames Feist if (inputs == nullptr) 5705b4aa86bSJames Feist { 571002d39b4SEd Tanous BMCWEB_LOG_ERROR << "Zones Pid Field Illegal"; 572a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 5735b4aa86bSJames Feist return; 5745b4aa86bSJames Feist } 575b7a08d04SJames Feist auto& data = (*config)[propertyPair.first]; 5765b4aa86bSJames Feist data = nlohmann::json::array(); 5775b4aa86bSJames Feist for (std::string itemCopy : *inputs) 5785b4aa86bSJames Feist { 5795b4aa86bSJames Feist dbus::utility::escapePathForDbus(itemCopy); 5801476687dSEd Tanous nlohmann::json::object_t input; 5811476687dSEd Tanous input["@odata.id"] = 5820fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/" + 5831476687dSEd Tanous itemCopy; 5841476687dSEd Tanous data.push_back(std::move(input)); 5855b4aa86bSJames Feist } 5865b4aa86bSJames Feist } 5875b4aa86bSJames Feist // todo(james): may never happen, but this 5885b4aa86bSJames Feist // assumes configuration data referenced in the 5895b4aa86bSJames Feist // PID config is provided by the same daemon, we 5905b4aa86bSJames Feist // could add another loop to cover all cases, 5915b4aa86bSJames Feist // but I'm okay kicking this can down the road a 5925b4aa86bSJames Feist // bit 5935b4aa86bSJames Feist 5945b4aa86bSJames Feist else if (propertyPair.first == "Inputs" || 5955b4aa86bSJames Feist propertyPair.first == "Outputs") 5965b4aa86bSJames Feist { 597b7a08d04SJames Feist auto& data = (*config)[propertyPair.first]; 5985b4aa86bSJames Feist const std::vector<std::string>* inputs = 599abf2add6SEd Tanous std::get_if<std::vector<std::string>>( 6001b6b96c5SEd Tanous &propertyPair.second); 6015b4aa86bSJames Feist 6025b4aa86bSJames Feist if (inputs == nullptr) 6035b4aa86bSJames Feist { 6045b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 6055b4aa86bSJames Feist << propertyPair.first; 606f12894f8SJason M. Bills messages::internalError(asyncResp->res); 6075b4aa86bSJames Feist return; 6085b4aa86bSJames Feist } 6095b4aa86bSJames Feist data = *inputs; 610b943aaefSJames Feist } 611b943aaefSJames Feist else if (propertyPair.first == "SetPointOffset") 612b943aaefSJames Feist { 613b943aaefSJames Feist const std::string* ptr = 614002d39b4SEd Tanous std::get_if<std::string>(&propertyPair.second); 615b943aaefSJames Feist 616b943aaefSJames Feist if (ptr == nullptr) 617b943aaefSJames Feist { 618b943aaefSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 619b943aaefSJames Feist << propertyPair.first; 620b943aaefSJames Feist messages::internalError(asyncResp->res); 621b943aaefSJames Feist return; 622b943aaefSJames Feist } 623b943aaefSJames Feist // translate from dbus to redfish 624b943aaefSJames Feist if (*ptr == "WarningHigh") 625b943aaefSJames Feist { 626b943aaefSJames Feist (*config)["SetPointOffset"] = 627b943aaefSJames Feist "UpperThresholdNonCritical"; 628b943aaefSJames Feist } 629b943aaefSJames Feist else if (*ptr == "WarningLow") 630b943aaefSJames Feist { 631b943aaefSJames Feist (*config)["SetPointOffset"] = 632b943aaefSJames Feist "LowerThresholdNonCritical"; 633b943aaefSJames Feist } 634b943aaefSJames Feist else if (*ptr == "CriticalHigh") 635b943aaefSJames Feist { 636b943aaefSJames Feist (*config)["SetPointOffset"] = 637b943aaefSJames Feist "UpperThresholdCritical"; 638b943aaefSJames Feist } 639b943aaefSJames Feist else if (*ptr == "CriticalLow") 640b943aaefSJames Feist { 641b943aaefSJames Feist (*config)["SetPointOffset"] = 642b943aaefSJames Feist "LowerThresholdCritical"; 643b943aaefSJames Feist } 644b943aaefSJames Feist else 645b943aaefSJames Feist { 646002d39b4SEd Tanous BMCWEB_LOG_ERROR << "Value Illegal " << *ptr; 647b943aaefSJames Feist messages::internalError(asyncResp->res); 648b943aaefSJames Feist return; 649b943aaefSJames Feist } 650b943aaefSJames Feist } 651b943aaefSJames Feist // doubles 652002d39b4SEd Tanous else if (propertyPair.first == "FFGainCoefficient" || 6535b4aa86bSJames Feist propertyPair.first == "FFOffCoefficient" || 6545b4aa86bSJames Feist propertyPair.first == "ICoefficient" || 6555b4aa86bSJames Feist propertyPair.first == "ILimitMax" || 6565b4aa86bSJames Feist propertyPair.first == "ILimitMin" || 657002d39b4SEd Tanous propertyPair.first == "PositiveHysteresis" || 658002d39b4SEd Tanous propertyPair.first == "NegativeHysteresis" || 6595b4aa86bSJames Feist propertyPair.first == "OutLimitMax" || 6605b4aa86bSJames Feist propertyPair.first == "OutLimitMin" || 6615b4aa86bSJames Feist propertyPair.first == "PCoefficient" || 6627625cb81SJames Feist propertyPair.first == "SetPoint" || 6635b4aa86bSJames Feist propertyPair.first == "SlewNeg" || 6645b4aa86bSJames Feist propertyPair.first == "SlewPos") 6655b4aa86bSJames Feist { 6665b4aa86bSJames Feist const double* ptr = 667abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 6685b4aa86bSJames Feist if (ptr == nullptr) 6695b4aa86bSJames Feist { 6705b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 6715b4aa86bSJames Feist << propertyPair.first; 672f12894f8SJason M. Bills messages::internalError(asyncResp->res); 6735b4aa86bSJames Feist return; 6745b4aa86bSJames Feist } 675b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 6765b4aa86bSJames Feist } 6775b4aa86bSJames Feist } 6785b4aa86bSJames Feist } 6795b4aa86bSJames Feist } 6805b4aa86bSJames Feist } 6815b4aa86bSJames Feist }, 6825b4aa86bSJames Feist connection, path, objectManagerIface, "GetManagedObjects"); 6835b4aa86bSJames Feist } 684ca537928SJennifer Lee 68583ff9ab6SJames Feist enum class CreatePIDRet 68683ff9ab6SJames Feist { 68783ff9ab6SJames Feist fail, 68883ff9ab6SJames Feist del, 68983ff9ab6SJames Feist patch 69083ff9ab6SJames Feist }; 69183ff9ab6SJames Feist 6928d1b46d7Szhanghch05 inline bool 6938d1b46d7Szhanghch05 getZonesFromJsonReq(const std::shared_ptr<bmcweb::AsyncResp>& response, 6945f2caaefSJames Feist std::vector<nlohmann::json>& config, 6955f2caaefSJames Feist std::vector<std::string>& zones) 6965f2caaefSJames Feist { 697b6baeaa4SJames Feist if (config.empty()) 698b6baeaa4SJames Feist { 699b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Empty Zones"; 7001668ce6dSEd Tanous messages::propertyValueFormatError(response->res, "[]", "Zones"); 701b6baeaa4SJames Feist return false; 702b6baeaa4SJames Feist } 7035f2caaefSJames Feist for (auto& odata : config) 7045f2caaefSJames Feist { 7055f2caaefSJames Feist std::string path; 7065f2caaefSJames Feist if (!redfish::json_util::readJson(odata, response->res, "@odata.id", 7075f2caaefSJames Feist path)) 7085f2caaefSJames Feist { 7095f2caaefSJames Feist return false; 7105f2caaefSJames Feist } 7115f2caaefSJames Feist std::string input; 71261adbda3SJames Feist 71361adbda3SJames Feist // 8 below comes from 71461adbda3SJames Feist // /redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/Left 71561adbda3SJames Feist // 0 1 2 3 4 5 6 7 8 71661adbda3SJames Feist if (!dbus::utility::getNthStringFromPath(path, 8, input)) 7175f2caaefSJames Feist { 7185f2caaefSJames Feist BMCWEB_LOG_ERROR << "Got invalid path " << path; 7195f2caaefSJames Feist BMCWEB_LOG_ERROR << "Illegal Type Zones"; 7205f2caaefSJames Feist messages::propertyValueFormatError(response->res, odata.dump(), 7215f2caaefSJames Feist "Zones"); 7225f2caaefSJames Feist return false; 7235f2caaefSJames Feist } 724a170f275SEd Tanous std::replace(input.begin(), input.end(), '_', ' '); 7255f2caaefSJames Feist zones.emplace_back(std::move(input)); 7265f2caaefSJames Feist } 7275f2caaefSJames Feist return true; 7285f2caaefSJames Feist } 7295f2caaefSJames Feist 730711ac7a9SEd Tanous inline const dbus::utility::ManagedObjectType::value_type* 73173df0db0SJames Feist findChassis(const dbus::utility::ManagedObjectType& managedObj, 732b6baeaa4SJames Feist const std::string& value, std::string& chassis) 733b6baeaa4SJames Feist { 734b6baeaa4SJames Feist BMCWEB_LOG_DEBUG << "Find Chassis: " << value << "\n"; 735b6baeaa4SJames Feist 736a170f275SEd Tanous std::string escaped = value; 737a170f275SEd Tanous std::replace(escaped.begin(), escaped.end(), '_', ' '); 738b6baeaa4SJames Feist escaped = "/" + escaped; 739002d39b4SEd Tanous auto it = std::find_if(managedObj.begin(), managedObj.end(), 740002d39b4SEd Tanous [&escaped](const auto& obj) { 741b6baeaa4SJames Feist if (boost::algorithm::ends_with(obj.first.str, escaped)) 742b6baeaa4SJames Feist { 743b6baeaa4SJames Feist BMCWEB_LOG_DEBUG << "Matched " << obj.first.str << "\n"; 744b6baeaa4SJames Feist return true; 745b6baeaa4SJames Feist } 746b6baeaa4SJames Feist return false; 747b6baeaa4SJames Feist }); 748b6baeaa4SJames Feist 749b6baeaa4SJames Feist if (it == managedObj.end()) 750b6baeaa4SJames Feist { 75173df0db0SJames Feist return nullptr; 752b6baeaa4SJames Feist } 753b6baeaa4SJames Feist // 5 comes from <chassis-name> being the 5th element 754b6baeaa4SJames Feist // /xyz/openbmc_project/inventory/system/chassis/<chassis-name> 75573df0db0SJames Feist if (dbus::utility::getNthStringFromPath(it->first.str, 5, chassis)) 75673df0db0SJames Feist { 75773df0db0SJames Feist return &(*it); 75873df0db0SJames Feist } 75973df0db0SJames Feist 76073df0db0SJames Feist return nullptr; 761b6baeaa4SJames Feist } 762b6baeaa4SJames Feist 76323a21a1cSEd Tanous inline CreatePIDRet createPidInterface( 7648d1b46d7Szhanghch05 const std::shared_ptr<bmcweb::AsyncResp>& response, const std::string& type, 765b5a76932SEd Tanous const nlohmann::json::iterator& it, const std::string& path, 76683ff9ab6SJames Feist const dbus::utility::ManagedObjectType& managedObj, bool createNewObject, 767b9d36b47SEd Tanous dbus::utility::DBusPropertiesMap& output, std::string& chassis, 768b9d36b47SEd Tanous const std::string& profile) 76983ff9ab6SJames Feist { 77083ff9ab6SJames Feist 7715f2caaefSJames Feist // common deleter 772b6baeaa4SJames Feist if (it.value() == nullptr) 7735f2caaefSJames Feist { 7745f2caaefSJames Feist std::string iface; 7755f2caaefSJames Feist if (type == "PidControllers" || type == "FanControllers") 7765f2caaefSJames Feist { 7775f2caaefSJames Feist iface = pidConfigurationIface; 7785f2caaefSJames Feist } 7795f2caaefSJames Feist else if (type == "FanZones") 7805f2caaefSJames Feist { 7815f2caaefSJames Feist iface = pidZoneConfigurationIface; 7825f2caaefSJames Feist } 7835f2caaefSJames Feist else if (type == "StepwiseControllers") 7845f2caaefSJames Feist { 7855f2caaefSJames Feist iface = stepwiseConfigurationIface; 7865f2caaefSJames Feist } 7875f2caaefSJames Feist else 7885f2caaefSJames Feist { 789a0744d38SGunnar Mills BMCWEB_LOG_ERROR << "Illegal Type " << type; 7905f2caaefSJames Feist messages::propertyUnknown(response->res, type); 7915f2caaefSJames Feist return CreatePIDRet::fail; 7925f2caaefSJames Feist } 7936ee7f774SJames Feist 7946ee7f774SJames Feist BMCWEB_LOG_DEBUG << "del " << path << " " << iface << "\n"; 7955f2caaefSJames Feist // delete interface 7965f2caaefSJames Feist crow::connections::systemBus->async_method_call( 7975f2caaefSJames Feist [response, path](const boost::system::error_code ec) { 7985f2caaefSJames Feist if (ec) 7995f2caaefSJames Feist { 8005f2caaefSJames Feist BMCWEB_LOG_ERROR << "Error patching " << path << ": " << ec; 8015f2caaefSJames Feist messages::internalError(response->res); 802b6baeaa4SJames Feist return; 8035f2caaefSJames Feist } 804b6baeaa4SJames Feist messages::success(response->res); 8055f2caaefSJames Feist }, 8065f2caaefSJames Feist "xyz.openbmc_project.EntityManager", path, iface, "Delete"); 8075f2caaefSJames Feist return CreatePIDRet::del; 8085f2caaefSJames Feist } 8095f2caaefSJames Feist 810711ac7a9SEd Tanous const dbus::utility::ManagedObjectType::value_type* managedItem = nullptr; 811b6baeaa4SJames Feist if (!createNewObject) 812b6baeaa4SJames Feist { 813b6baeaa4SJames Feist // if we aren't creating a new object, we should be able to find it on 814b6baeaa4SJames Feist // d-bus 81573df0db0SJames Feist managedItem = findChassis(managedObj, it.key(), chassis); 81673df0db0SJames Feist if (managedItem == nullptr) 817b6baeaa4SJames Feist { 818b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 819ace85d60SEd Tanous messages::invalidObject(response->res, 820ace85d60SEd Tanous crow::utility::urlFromPieces( 821ace85d60SEd Tanous "redfish", "v1", "Chassis", chassis)); 822b6baeaa4SJames Feist return CreatePIDRet::fail; 823b6baeaa4SJames Feist } 824b6baeaa4SJames Feist } 825b6baeaa4SJames Feist 82626f6976fSEd Tanous if (!profile.empty() && 82773df0db0SJames Feist (type == "PidControllers" || type == "FanControllers" || 82873df0db0SJames Feist type == "StepwiseControllers")) 82973df0db0SJames Feist { 83073df0db0SJames Feist if (managedItem == nullptr) 83173df0db0SJames Feist { 832b9d36b47SEd Tanous output.emplace_back("Profiles", std::vector<std::string>{profile}); 83373df0db0SJames Feist } 83473df0db0SJames Feist else 83573df0db0SJames Feist { 83673df0db0SJames Feist std::string interface; 83773df0db0SJames Feist if (type == "StepwiseControllers") 83873df0db0SJames Feist { 83973df0db0SJames Feist interface = stepwiseConfigurationIface; 84073df0db0SJames Feist } 84173df0db0SJames Feist else 84273df0db0SJames Feist { 84373df0db0SJames Feist interface = pidConfigurationIface; 84473df0db0SJames Feist } 845711ac7a9SEd Tanous bool ifaceFound = false; 846711ac7a9SEd Tanous for (const auto& iface : managedItem->second) 847711ac7a9SEd Tanous { 848711ac7a9SEd Tanous if (iface.first == interface) 849711ac7a9SEd Tanous { 850711ac7a9SEd Tanous ifaceFound = true; 851711ac7a9SEd Tanous for (const auto& prop : iface.second) 852711ac7a9SEd Tanous { 853711ac7a9SEd Tanous if (prop.first == "Profiles") 854711ac7a9SEd Tanous { 855711ac7a9SEd Tanous const std::vector<std::string>* curProfiles = 856711ac7a9SEd Tanous std::get_if<std::vector<std::string>>( 857711ac7a9SEd Tanous &(prop.second)); 858711ac7a9SEd Tanous if (curProfiles == nullptr) 859711ac7a9SEd Tanous { 860711ac7a9SEd Tanous BMCWEB_LOG_ERROR 861711ac7a9SEd Tanous << "Illegal profiles in managed object"; 862711ac7a9SEd Tanous messages::internalError(response->res); 863711ac7a9SEd Tanous return CreatePIDRet::fail; 864711ac7a9SEd Tanous } 865711ac7a9SEd Tanous if (std::find(curProfiles->begin(), 866711ac7a9SEd Tanous curProfiles->end(), 867711ac7a9SEd Tanous profile) == curProfiles->end()) 868711ac7a9SEd Tanous { 869711ac7a9SEd Tanous std::vector<std::string> newProfiles = 870711ac7a9SEd Tanous *curProfiles; 871711ac7a9SEd Tanous newProfiles.push_back(profile); 872b9d36b47SEd Tanous output.emplace_back("Profiles", newProfiles); 873711ac7a9SEd Tanous } 874711ac7a9SEd Tanous } 875711ac7a9SEd Tanous } 876711ac7a9SEd Tanous } 877711ac7a9SEd Tanous } 878711ac7a9SEd Tanous 879711ac7a9SEd Tanous if (!ifaceFound) 88073df0db0SJames Feist { 88173df0db0SJames Feist BMCWEB_LOG_ERROR 88273df0db0SJames Feist << "Failed to find interface in managed object"; 88373df0db0SJames Feist messages::internalError(response->res); 88473df0db0SJames Feist return CreatePIDRet::fail; 88573df0db0SJames Feist } 88673df0db0SJames Feist } 88773df0db0SJames Feist } 88873df0db0SJames Feist 88983ff9ab6SJames Feist if (type == "PidControllers" || type == "FanControllers") 89083ff9ab6SJames Feist { 89183ff9ab6SJames Feist if (createNewObject) 89283ff9ab6SJames Feist { 893b9d36b47SEd Tanous output.emplace_back("Class", 894b9d36b47SEd Tanous type == "PidControllers" ? "temp" : "fan"); 895b9d36b47SEd Tanous output.emplace_back("Type", "Pid"); 89683ff9ab6SJames Feist } 8975f2caaefSJames Feist 8985f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> zones; 8995f2caaefSJames Feist std::optional<std::vector<std::string>> inputs; 9005f2caaefSJames Feist std::optional<std::vector<std::string>> outputs; 9015f2caaefSJames Feist std::map<std::string, std::optional<double>> doubles; 902b943aaefSJames Feist std::optional<std::string> setpointOffset; 9035f2caaefSJames Feist if (!redfish::json_util::readJson( 904b6baeaa4SJames Feist it.value(), response->res, "Inputs", inputs, "Outputs", outputs, 9055f2caaefSJames Feist "Zones", zones, "FFGainCoefficient", 9065f2caaefSJames Feist doubles["FFGainCoefficient"], "FFOffCoefficient", 9075f2caaefSJames Feist doubles["FFOffCoefficient"], "ICoefficient", 9085f2caaefSJames Feist doubles["ICoefficient"], "ILimitMax", doubles["ILimitMax"], 9095f2caaefSJames Feist "ILimitMin", doubles["ILimitMin"], "OutLimitMax", 9105f2caaefSJames Feist doubles["OutLimitMax"], "OutLimitMin", doubles["OutLimitMin"], 9115f2caaefSJames Feist "PCoefficient", doubles["PCoefficient"], "SetPoint", 912b943aaefSJames Feist doubles["SetPoint"], "SetPointOffset", setpointOffset, 913b943aaefSJames Feist "SlewNeg", doubles["SlewNeg"], "SlewPos", doubles["SlewPos"], 914b943aaefSJames Feist "PositiveHysteresis", doubles["PositiveHysteresis"], 915b943aaefSJames Feist "NegativeHysteresis", doubles["NegativeHysteresis"])) 91683ff9ab6SJames Feist { 91771f52d96SEd Tanous BMCWEB_LOG_ERROR 91871f52d96SEd Tanous << "Illegal Property " 91971f52d96SEd Tanous << it.value().dump(2, ' ', true, 92071f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 9215f2caaefSJames Feist return CreatePIDRet::fail; 92283ff9ab6SJames Feist } 9235f2caaefSJames Feist if (zones) 9245f2caaefSJames Feist { 9255f2caaefSJames Feist std::vector<std::string> zonesStr; 9265f2caaefSJames Feist if (!getZonesFromJsonReq(response, *zones, zonesStr)) 9275f2caaefSJames Feist { 928a0744d38SGunnar Mills BMCWEB_LOG_ERROR << "Illegal Zones"; 9295f2caaefSJames Feist return CreatePIDRet::fail; 9305f2caaefSJames Feist } 931b6baeaa4SJames Feist if (chassis.empty() && 932e662eae8SEd Tanous findChassis(managedObj, zonesStr[0], chassis) == nullptr) 933b6baeaa4SJames Feist { 934b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 935ace85d60SEd Tanous messages::invalidObject( 936ace85d60SEd Tanous response->res, crow::utility::urlFromPieces( 937ace85d60SEd Tanous "redfish", "v1", "Chassis", chassis)); 938b6baeaa4SJames Feist return CreatePIDRet::fail; 939b6baeaa4SJames Feist } 940b9d36b47SEd Tanous output.emplace_back("Zones", std::move(zonesStr)); 9415f2caaefSJames Feist } 942afb9ee06SEd Tanous 943afb9ee06SEd Tanous if (inputs) 9445f2caaefSJames Feist { 945afb9ee06SEd Tanous for (std::string& value : *inputs) 94683ff9ab6SJames Feist { 947a170f275SEd Tanous std::replace(value.begin(), value.end(), '_', ' '); 94883ff9ab6SJames Feist } 949afb9ee06SEd Tanous output.emplace_back("Inputs", *inputs); 950afb9ee06SEd Tanous } 951afb9ee06SEd Tanous 952afb9ee06SEd Tanous if (outputs) 9535f2caaefSJames Feist { 954afb9ee06SEd Tanous for (std::string& value : *outputs) 9555f2caaefSJames Feist { 956afb9ee06SEd Tanous std::replace(value.begin(), value.end(), '_', ' '); 9575f2caaefSJames Feist } 958afb9ee06SEd Tanous output.emplace_back("Outputs", *outputs); 95983ff9ab6SJames Feist } 96083ff9ab6SJames Feist 961b943aaefSJames Feist if (setpointOffset) 962b943aaefSJames Feist { 963b943aaefSJames Feist // translate between redfish and dbus names 964b943aaefSJames Feist if (*setpointOffset == "UpperThresholdNonCritical") 965b943aaefSJames Feist { 966b9d36b47SEd Tanous output.emplace_back("SetPointOffset", "WarningLow"); 967b943aaefSJames Feist } 968b943aaefSJames Feist else if (*setpointOffset == "LowerThresholdNonCritical") 969b943aaefSJames Feist { 970b9d36b47SEd Tanous output.emplace_back("SetPointOffset", "WarningHigh"); 971b943aaefSJames Feist } 972b943aaefSJames Feist else if (*setpointOffset == "LowerThresholdCritical") 973b943aaefSJames Feist { 974b9d36b47SEd Tanous output.emplace_back("SetPointOffset", "CriticalLow"); 975b943aaefSJames Feist } 976b943aaefSJames Feist else if (*setpointOffset == "UpperThresholdCritical") 977b943aaefSJames Feist { 978b9d36b47SEd Tanous output.emplace_back("SetPointOffset", "CriticalHigh"); 979b943aaefSJames Feist } 980b943aaefSJames Feist else 981b943aaefSJames Feist { 982b943aaefSJames Feist BMCWEB_LOG_ERROR << "Invalid setpointoffset " 983b943aaefSJames Feist << *setpointOffset; 984ace85d60SEd Tanous messages::propertyValueNotInList(response->res, it.key(), 985ace85d60SEd Tanous "SetPointOffset"); 986b943aaefSJames Feist return CreatePIDRet::fail; 987b943aaefSJames Feist } 988b943aaefSJames Feist } 989b943aaefSJames Feist 99083ff9ab6SJames Feist // doubles 9915f2caaefSJames Feist for (const auto& pairs : doubles) 99283ff9ab6SJames Feist { 9935f2caaefSJames Feist if (!pairs.second) 99483ff9ab6SJames Feist { 9955f2caaefSJames Feist continue; 99683ff9ab6SJames Feist } 9975f2caaefSJames Feist BMCWEB_LOG_DEBUG << pairs.first << " = " << *pairs.second; 998b9d36b47SEd Tanous output.emplace_back(pairs.first, *pairs.second); 9995f2caaefSJames Feist } 100083ff9ab6SJames Feist } 100183ff9ab6SJames Feist 100283ff9ab6SJames Feist else if (type == "FanZones") 100383ff9ab6SJames Feist { 1004b9d36b47SEd Tanous output.emplace_back("Type", "Pid.Zone"); 100583ff9ab6SJames Feist 10065f2caaefSJames Feist std::optional<nlohmann::json> chassisContainer; 10075f2caaefSJames Feist std::optional<double> failSafePercent; 1008d3ec07f8SJames Feist std::optional<double> minThermalOutput; 1009b6baeaa4SJames Feist if (!redfish::json_util::readJson(it.value(), response->res, "Chassis", 10105f2caaefSJames Feist chassisContainer, "FailSafePercent", 1011d3ec07f8SJames Feist failSafePercent, "MinThermalOutput", 1012d3ec07f8SJames Feist minThermalOutput)) 101383ff9ab6SJames Feist { 101471f52d96SEd Tanous BMCWEB_LOG_ERROR 101571f52d96SEd Tanous << "Illegal Property " 101671f52d96SEd Tanous << it.value().dump(2, ' ', true, 101771f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 101883ff9ab6SJames Feist return CreatePIDRet::fail; 101983ff9ab6SJames Feist } 10205f2caaefSJames Feist 10215f2caaefSJames Feist if (chassisContainer) 102283ff9ab6SJames Feist { 10235f2caaefSJames Feist 10245f2caaefSJames Feist std::string chassisId; 10255f2caaefSJames Feist if (!redfish::json_util::readJson(*chassisContainer, response->res, 10265f2caaefSJames Feist "@odata.id", chassisId)) 10275f2caaefSJames Feist { 102871f52d96SEd Tanous BMCWEB_LOG_ERROR 102971f52d96SEd Tanous << "Illegal Property " 103071f52d96SEd Tanous << chassisContainer->dump( 103171f52d96SEd Tanous 2, ' ', true, 103271f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 103383ff9ab6SJames Feist return CreatePIDRet::fail; 103483ff9ab6SJames Feist } 103583ff9ab6SJames Feist 1036717794d5SAppaRao Puli // /redfish/v1/chassis/chassis_name/ 10375f2caaefSJames Feist if (!dbus::utility::getNthStringFromPath(chassisId, 3, chassis)) 103883ff9ab6SJames Feist { 10395f2caaefSJames Feist BMCWEB_LOG_ERROR << "Got invalid path " << chassisId; 1040ace85d60SEd Tanous messages::invalidObject( 1041ace85d60SEd Tanous response->res, crow::utility::urlFromPieces( 1042ace85d60SEd Tanous "redfish", "v1", "Chassis", chassisId)); 104383ff9ab6SJames Feist return CreatePIDRet::fail; 104483ff9ab6SJames Feist } 104583ff9ab6SJames Feist } 1046d3ec07f8SJames Feist if (minThermalOutput) 104783ff9ab6SJames Feist { 1048b9d36b47SEd Tanous output.emplace_back("MinThermalOutput", *minThermalOutput); 10495f2caaefSJames Feist } 10505f2caaefSJames Feist if (failSafePercent) 105183ff9ab6SJames Feist { 1052b9d36b47SEd Tanous output.emplace_back("FailSafePercent", *failSafePercent); 10535f2caaefSJames Feist } 10545f2caaefSJames Feist } 10555f2caaefSJames Feist else if (type == "StepwiseControllers") 10565f2caaefSJames Feist { 1057b9d36b47SEd Tanous output.emplace_back("Type", "Stepwise"); 10585f2caaefSJames Feist 10595f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> zones; 10605f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> steps; 10615f2caaefSJames Feist std::optional<std::vector<std::string>> inputs; 10625f2caaefSJames Feist std::optional<double> positiveHysteresis; 10635f2caaefSJames Feist std::optional<double> negativeHysteresis; 1064c33a90ecSJames Feist std::optional<std::string> direction; // upper clipping curve vs lower 10655f2caaefSJames Feist if (!redfish::json_util::readJson( 1066b6baeaa4SJames Feist it.value(), response->res, "Zones", zones, "Steps", steps, 1067b6baeaa4SJames Feist "Inputs", inputs, "PositiveHysteresis", positiveHysteresis, 1068c33a90ecSJames Feist "NegativeHysteresis", negativeHysteresis, "Direction", 1069c33a90ecSJames Feist direction)) 10705f2caaefSJames Feist { 107171f52d96SEd Tanous BMCWEB_LOG_ERROR 107271f52d96SEd Tanous << "Illegal Property " 107371f52d96SEd Tanous << it.value().dump(2, ' ', true, 107471f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 107583ff9ab6SJames Feist return CreatePIDRet::fail; 107683ff9ab6SJames Feist } 10775f2caaefSJames Feist 10785f2caaefSJames Feist if (zones) 107983ff9ab6SJames Feist { 1080b6baeaa4SJames Feist std::vector<std::string> zonesStrs; 1081b6baeaa4SJames Feist if (!getZonesFromJsonReq(response, *zones, zonesStrs)) 10825f2caaefSJames Feist { 1083a0744d38SGunnar Mills BMCWEB_LOG_ERROR << "Illegal Zones"; 108483ff9ab6SJames Feist return CreatePIDRet::fail; 108583ff9ab6SJames Feist } 1086b6baeaa4SJames Feist if (chassis.empty() && 1087e662eae8SEd Tanous findChassis(managedObj, zonesStrs[0], chassis) == nullptr) 1088b6baeaa4SJames Feist { 1089b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 1090ace85d60SEd Tanous messages::invalidObject( 1091ace85d60SEd Tanous response->res, crow::utility::urlFromPieces( 1092ace85d60SEd Tanous "redfish", "v1", "Chassis", chassis)); 1093b6baeaa4SJames Feist return CreatePIDRet::fail; 1094b6baeaa4SJames Feist } 1095b9d36b47SEd Tanous output.emplace_back("Zones", std::move(zonesStrs)); 10965f2caaefSJames Feist } 10975f2caaefSJames Feist if (steps) 10985f2caaefSJames Feist { 10995f2caaefSJames Feist std::vector<double> readings; 11005f2caaefSJames Feist std::vector<double> outputs; 11015f2caaefSJames Feist for (auto& step : *steps) 11025f2caaefSJames Feist { 1103543f4400SEd Tanous double target = 0.0; 1104543f4400SEd Tanous double out = 0.0; 11055f2caaefSJames Feist 11065f2caaefSJames Feist if (!redfish::json_util::readJson(step, response->res, "Target", 110723a21a1cSEd Tanous target, "Output", out)) 11085f2caaefSJames Feist { 110971f52d96SEd Tanous BMCWEB_LOG_ERROR 111071f52d96SEd Tanous << "Illegal Property " 111171f52d96SEd Tanous << it.value().dump( 111271f52d96SEd Tanous 2, ' ', true, 111371f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 11145f2caaefSJames Feist return CreatePIDRet::fail; 11155f2caaefSJames Feist } 11165f2caaefSJames Feist readings.emplace_back(target); 111723a21a1cSEd Tanous outputs.emplace_back(out); 11185f2caaefSJames Feist } 1119b9d36b47SEd Tanous output.emplace_back("Reading", std::move(readings)); 1120b9d36b47SEd Tanous output.emplace_back("Output", std::move(outputs)); 11215f2caaefSJames Feist } 11225f2caaefSJames Feist if (inputs) 11235f2caaefSJames Feist { 11245f2caaefSJames Feist for (std::string& value : *inputs) 11255f2caaefSJames Feist { 1126a170f275SEd Tanous 1127a170f275SEd Tanous std::replace(value.begin(), value.end(), '_', ' '); 11285f2caaefSJames Feist } 1129b9d36b47SEd Tanous output.emplace_back("Inputs", std::move(*inputs)); 11305f2caaefSJames Feist } 11315f2caaefSJames Feist if (negativeHysteresis) 11325f2caaefSJames Feist { 1133b9d36b47SEd Tanous output.emplace_back("NegativeHysteresis", *negativeHysteresis); 11345f2caaefSJames Feist } 11355f2caaefSJames Feist if (positiveHysteresis) 11365f2caaefSJames Feist { 1137b9d36b47SEd Tanous output.emplace_back("PositiveHysteresis", *positiveHysteresis); 113883ff9ab6SJames Feist } 1139c33a90ecSJames Feist if (direction) 1140c33a90ecSJames Feist { 1141c33a90ecSJames Feist constexpr const std::array<const char*, 2> allowedDirections = { 1142c33a90ecSJames Feist "Ceiling", "Floor"}; 1143c33a90ecSJames Feist if (std::find(allowedDirections.begin(), allowedDirections.end(), 1144c33a90ecSJames Feist *direction) == allowedDirections.end()) 1145c33a90ecSJames Feist { 1146c33a90ecSJames Feist messages::propertyValueTypeError(response->res, "Direction", 1147c33a90ecSJames Feist *direction); 1148c33a90ecSJames Feist return CreatePIDRet::fail; 1149c33a90ecSJames Feist } 1150b9d36b47SEd Tanous output.emplace_back("Class", *direction); 1151c33a90ecSJames Feist } 115283ff9ab6SJames Feist } 115383ff9ab6SJames Feist else 115483ff9ab6SJames Feist { 1155a0744d38SGunnar Mills BMCWEB_LOG_ERROR << "Illegal Type " << type; 115635a62c7cSJason M. Bills messages::propertyUnknown(response->res, type); 115783ff9ab6SJames Feist return CreatePIDRet::fail; 115883ff9ab6SJames Feist } 115983ff9ab6SJames Feist return CreatePIDRet::patch; 116083ff9ab6SJames Feist } 116173df0db0SJames Feist struct GetPIDValues : std::enable_shared_from_this<GetPIDValues> 116273df0db0SJames Feist { 11636936afe4SEd Tanous struct CompletionValues 11646936afe4SEd Tanous { 11656936afe4SEd Tanous std::vector<std::string> supportedProfiles; 11666936afe4SEd Tanous std::string currentProfile; 11676936afe4SEd Tanous dbus::utility::MapperGetSubTreeResponse subtree; 11686936afe4SEd Tanous }; 116983ff9ab6SJames Feist 11704e23a444SEd Tanous explicit GetPIDValues( 11714e23a444SEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn) : 117223a21a1cSEd Tanous asyncResp(asyncRespIn) 117373df0db0SJames Feist 11741214b7e7SGunnar Mills {} 11759c310685SBorawski.Lukasz 117673df0db0SJames Feist void run() 11775b4aa86bSJames Feist { 117873df0db0SJames Feist std::shared_ptr<GetPIDValues> self = shared_from_this(); 117973df0db0SJames Feist 118073df0db0SJames Feist // get all configurations 11815b4aa86bSJames Feist crow::connections::systemBus->async_method_call( 1182b9d36b47SEd Tanous [self]( 1183b9d36b47SEd Tanous const boost::system::error_code ec, 1184b9d36b47SEd Tanous const dbus::utility::MapperGetSubTreeResponse& subtreeLocal) { 11855b4aa86bSJames Feist if (ec) 11865b4aa86bSJames Feist { 11875b4aa86bSJames Feist BMCWEB_LOG_ERROR << ec; 118873df0db0SJames Feist messages::internalError(self->asyncResp->res); 118973df0db0SJames Feist return; 119073df0db0SJames Feist } 11916936afe4SEd Tanous self->complete.subtree = subtreeLocal; 119273df0db0SJames Feist }, 119373df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", 119473df0db0SJames Feist "/xyz/openbmc_project/object_mapper", 119573df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 119673df0db0SJames Feist std::array<const char*, 4>{ 119773df0db0SJames Feist pidConfigurationIface, pidZoneConfigurationIface, 119873df0db0SJames Feist objectManagerIface, stepwiseConfigurationIface}); 119973df0db0SJames Feist 120073df0db0SJames Feist // at the same time get the selected profile 120173df0db0SJames Feist crow::connections::systemBus->async_method_call( 1202b9d36b47SEd Tanous [self]( 1203b9d36b47SEd Tanous const boost::system::error_code ec, 1204b9d36b47SEd Tanous const dbus::utility::MapperGetSubTreeResponse& subtreeLocal) { 120523a21a1cSEd Tanous if (ec || subtreeLocal.empty()) 120673df0db0SJames Feist { 120773df0db0SJames Feist return; 120873df0db0SJames Feist } 120923a21a1cSEd Tanous if (subtreeLocal[0].second.size() != 1) 121073df0db0SJames Feist { 121173df0db0SJames Feist // invalid mapper response, should never happen 121273df0db0SJames Feist BMCWEB_LOG_ERROR << "GetPIDValues: Mapper Error"; 121373df0db0SJames Feist messages::internalError(self->asyncResp->res); 12145b4aa86bSJames Feist return; 12155b4aa86bSJames Feist } 12165b4aa86bSJames Feist 121723a21a1cSEd Tanous const std::string& path = subtreeLocal[0].first; 121823a21a1cSEd Tanous const std::string& owner = subtreeLocal[0].second[0].first; 1219fac6e53bSKrzysztof Grobelny 1220fac6e53bSKrzysztof Grobelny sdbusplus::asio::getAllProperties( 1221fac6e53bSKrzysztof Grobelny *crow::connections::systemBus, owner, path, thermalModeIface, 1222168e20c1SEd Tanous [path, owner, 1223168e20c1SEd Tanous self](const boost::system::error_code ec2, 1224b9d36b47SEd Tanous const dbus::utility::DBusPropertiesMap& resp) { 122523a21a1cSEd Tanous if (ec2) 122673df0db0SJames Feist { 12270fda0f12SGeorge Liu BMCWEB_LOG_ERROR 1228002d39b4SEd Tanous << "GetPIDValues: Can't get thermalModeIface " << path; 122973df0db0SJames Feist messages::internalError(self->asyncResp->res); 123073df0db0SJames Feist return; 123173df0db0SJames Feist } 1232fac6e53bSKrzysztof Grobelny 1233271584abSEd Tanous const std::string* current = nullptr; 1234271584abSEd Tanous const std::vector<std::string>* supported = nullptr; 1235fac6e53bSKrzysztof Grobelny 1236fac6e53bSKrzysztof Grobelny const bool success = sdbusplus::unpackPropertiesNoThrow( 1237fac6e53bSKrzysztof Grobelny dbus_utils::UnpackErrorPrinter(), resp, "Current", current, 1238fac6e53bSKrzysztof Grobelny "Supported", supported); 1239fac6e53bSKrzysztof Grobelny 1240fac6e53bSKrzysztof Grobelny if (!success) 124173df0db0SJames Feist { 1242002d39b4SEd Tanous messages::internalError(self->asyncResp->res); 124373df0db0SJames Feist return; 124473df0db0SJames Feist } 1245fac6e53bSKrzysztof Grobelny 124673df0db0SJames Feist if (current == nullptr || supported == nullptr) 124773df0db0SJames Feist { 12480fda0f12SGeorge Liu BMCWEB_LOG_ERROR 1249002d39b4SEd Tanous << "GetPIDValues: thermal mode iface invalid " << path; 125073df0db0SJames Feist messages::internalError(self->asyncResp->res); 125173df0db0SJames Feist return; 125273df0db0SJames Feist } 12536936afe4SEd Tanous self->complete.currentProfile = *current; 12546936afe4SEd Tanous self->complete.supportedProfiles = *supported; 1255fac6e53bSKrzysztof Grobelny }); 125673df0db0SJames Feist }, 125773df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", 125873df0db0SJames Feist "/xyz/openbmc_project/object_mapper", 125973df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 126073df0db0SJames Feist std::array<const char*, 1>{thermalModeIface}); 126173df0db0SJames Feist } 126273df0db0SJames Feist 12636936afe4SEd Tanous static void 12646936afe4SEd Tanous processingComplete(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 12656936afe4SEd Tanous const CompletionValues& completion) 126673df0db0SJames Feist { 126773df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 126873df0db0SJames Feist { 126973df0db0SJames Feist return; 127073df0db0SJames Feist } 12715b4aa86bSJames Feist // create map of <connection, path to objMgr>> 12726936afe4SEd Tanous boost::container::flat_map< 12736936afe4SEd Tanous std::string, std::string, std::less<>, 12746936afe4SEd Tanous std::vector<std::pair<std::string, std::string>>> 12756936afe4SEd Tanous objectMgrPaths; 12766936afe4SEd Tanous boost::container::flat_set<std::string, std::less<>, 12776936afe4SEd Tanous std::vector<std::string>> 12786936afe4SEd Tanous calledConnections; 12796936afe4SEd Tanous for (const auto& pathGroup : completion.subtree) 12805b4aa86bSJames Feist { 12815b4aa86bSJames Feist for (const auto& connectionGroup : pathGroup.second) 12825b4aa86bSJames Feist { 12836bce33bcSJames Feist auto findConnection = 12846bce33bcSJames Feist calledConnections.find(connectionGroup.first); 12856bce33bcSJames Feist if (findConnection != calledConnections.end()) 12866bce33bcSJames Feist { 12876bce33bcSJames Feist break; 12886bce33bcSJames Feist } 128973df0db0SJames Feist for (const std::string& interface : connectionGroup.second) 12905b4aa86bSJames Feist { 12915b4aa86bSJames Feist if (interface == objectManagerIface) 12925b4aa86bSJames Feist { 129373df0db0SJames Feist objectMgrPaths[connectionGroup.first] = pathGroup.first; 12945b4aa86bSJames Feist } 12955b4aa86bSJames Feist // this list is alphabetical, so we 12965b4aa86bSJames Feist // should have found the objMgr by now 12975b4aa86bSJames Feist if (interface == pidConfigurationIface || 1298b7a08d04SJames Feist interface == pidZoneConfigurationIface || 1299b7a08d04SJames Feist interface == stepwiseConfigurationIface) 13005b4aa86bSJames Feist { 13015b4aa86bSJames Feist auto findObjMgr = 13025b4aa86bSJames Feist objectMgrPaths.find(connectionGroup.first); 13035b4aa86bSJames Feist if (findObjMgr == objectMgrPaths.end()) 13045b4aa86bSJames Feist { 13055b4aa86bSJames Feist BMCWEB_LOG_DEBUG << connectionGroup.first 13065b4aa86bSJames Feist << "Has no Object Manager"; 13075b4aa86bSJames Feist continue; 13085b4aa86bSJames Feist } 13096bce33bcSJames Feist 13106bce33bcSJames Feist calledConnections.insert(connectionGroup.first); 13116bce33bcSJames Feist 131273df0db0SJames Feist asyncPopulatePid(findObjMgr->first, findObjMgr->second, 13136936afe4SEd Tanous completion.currentProfile, 13146936afe4SEd Tanous completion.supportedProfiles, 131573df0db0SJames Feist asyncResp); 13165b4aa86bSJames Feist break; 13175b4aa86bSJames Feist } 13185b4aa86bSJames Feist } 13195b4aa86bSJames Feist } 13205b4aa86bSJames Feist } 132173df0db0SJames Feist } 132273df0db0SJames Feist 13236936afe4SEd Tanous ~GetPIDValues() 13246936afe4SEd Tanous { 13256936afe4SEd Tanous boost::asio::post(crow::connections::systemBus->get_io_context(), 13266936afe4SEd Tanous std::bind_front(&processingComplete, asyncResp, 13276936afe4SEd Tanous std::move(complete))); 13286936afe4SEd Tanous } 13296936afe4SEd Tanous 1330ecd6a3a2SEd Tanous GetPIDValues(const GetPIDValues&) = delete; 1331ecd6a3a2SEd Tanous GetPIDValues(GetPIDValues&&) = delete; 1332ecd6a3a2SEd Tanous GetPIDValues& operator=(const GetPIDValues&) = delete; 1333ecd6a3a2SEd Tanous GetPIDValues& operator=(GetPIDValues&&) = delete; 1334ecd6a3a2SEd Tanous 13358d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> asyncResp; 13366936afe4SEd Tanous CompletionValues complete; 133773df0db0SJames Feist }; 133873df0db0SJames Feist 133973df0db0SJames Feist struct SetPIDValues : std::enable_shared_from_this<SetPIDValues> 134073df0db0SJames Feist { 134173df0db0SJames Feist 13428d1b46d7Szhanghch05 SetPIDValues(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn, 134373df0db0SJames Feist nlohmann::json& data) : 1344271584abSEd Tanous asyncResp(asyncRespIn) 134573df0db0SJames Feist { 134673df0db0SJames Feist 134773df0db0SJames Feist std::optional<nlohmann::json> pidControllers; 134873df0db0SJames Feist std::optional<nlohmann::json> fanControllers; 134973df0db0SJames Feist std::optional<nlohmann::json> fanZones; 135073df0db0SJames Feist std::optional<nlohmann::json> stepwiseControllers; 135173df0db0SJames Feist 135273df0db0SJames Feist if (!redfish::json_util::readJson( 135373df0db0SJames Feist data, asyncResp->res, "PidControllers", pidControllers, 135473df0db0SJames Feist "FanControllers", fanControllers, "FanZones", fanZones, 135573df0db0SJames Feist "StepwiseControllers", stepwiseControllers, "Profile", profile)) 135673df0db0SJames Feist { 135771f52d96SEd Tanous BMCWEB_LOG_ERROR 135871f52d96SEd Tanous << "Illegal Property " 135971f52d96SEd Tanous << data.dump(2, ' ', true, 136071f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 136173df0db0SJames Feist return; 136273df0db0SJames Feist } 136373df0db0SJames Feist configuration.emplace_back("PidControllers", std::move(pidControllers)); 136473df0db0SJames Feist configuration.emplace_back("FanControllers", std::move(fanControllers)); 136573df0db0SJames Feist configuration.emplace_back("FanZones", std::move(fanZones)); 136673df0db0SJames Feist configuration.emplace_back("StepwiseControllers", 136773df0db0SJames Feist std::move(stepwiseControllers)); 136873df0db0SJames Feist } 1369ecd6a3a2SEd Tanous 1370ecd6a3a2SEd Tanous SetPIDValues(const SetPIDValues&) = delete; 1371ecd6a3a2SEd Tanous SetPIDValues(SetPIDValues&&) = delete; 1372ecd6a3a2SEd Tanous SetPIDValues& operator=(const SetPIDValues&) = delete; 1373ecd6a3a2SEd Tanous SetPIDValues& operator=(SetPIDValues&&) = delete; 1374ecd6a3a2SEd Tanous 137573df0db0SJames Feist void run() 137673df0db0SJames Feist { 137773df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 137873df0db0SJames Feist { 137973df0db0SJames Feist return; 138073df0db0SJames Feist } 138173df0db0SJames Feist 138273df0db0SJames Feist std::shared_ptr<SetPIDValues> self = shared_from_this(); 138373df0db0SJames Feist 138473df0db0SJames Feist // todo(james): might make sense to do a mapper call here if this 138573df0db0SJames Feist // interface gets more traction 138673df0db0SJames Feist crow::connections::systemBus->async_method_call( 138773df0db0SJames Feist [self](const boost::system::error_code ec, 1388914e2d5dSEd Tanous const dbus::utility::ManagedObjectType& mObj) { 138973df0db0SJames Feist if (ec) 139073df0db0SJames Feist { 139173df0db0SJames Feist BMCWEB_LOG_ERROR << "Error communicating to Entity Manager"; 139273df0db0SJames Feist messages::internalError(self->asyncResp->res); 139373df0db0SJames Feist return; 139473df0db0SJames Feist } 1395e69d9de2SJames Feist const std::array<const char*, 3> configurations = { 1396e69d9de2SJames Feist pidConfigurationIface, pidZoneConfigurationIface, 1397e69d9de2SJames Feist stepwiseConfigurationIface}; 1398e69d9de2SJames Feist 139914b0b8d5SJames Feist for (const auto& [path, object] : mObj) 1400e69d9de2SJames Feist { 140114b0b8d5SJames Feist for (const auto& [interface, _] : object) 1402e69d9de2SJames Feist { 1403002d39b4SEd Tanous if (std::find(configurations.begin(), configurations.end(), 1404e69d9de2SJames Feist interface) != configurations.end()) 1405e69d9de2SJames Feist { 140614b0b8d5SJames Feist self->objectCount++; 1407e69d9de2SJames Feist break; 1408e69d9de2SJames Feist } 1409e69d9de2SJames Feist } 1410e69d9de2SJames Feist } 1411914e2d5dSEd Tanous self->managedObj = mObj; 141273df0db0SJames Feist }, 1413c106b67aSNan Zhou "xyz.openbmc_project.EntityManager", 1414c106b67aSNan Zhou "/xyz/openbmc_project/inventory", objectManagerIface, 141573df0db0SJames Feist "GetManagedObjects"); 141673df0db0SJames Feist 141773df0db0SJames Feist // at the same time get the profile information 141873df0db0SJames Feist crow::connections::systemBus->async_method_call( 141973df0db0SJames Feist [self](const boost::system::error_code ec, 1420b9d36b47SEd Tanous const dbus::utility::MapperGetSubTreeResponse& subtree) { 142173df0db0SJames Feist if (ec || subtree.empty()) 142273df0db0SJames Feist { 142373df0db0SJames Feist return; 142473df0db0SJames Feist } 142573df0db0SJames Feist if (subtree[0].second.empty()) 142673df0db0SJames Feist { 142773df0db0SJames Feist // invalid mapper response, should never happen 142873df0db0SJames Feist BMCWEB_LOG_ERROR << "SetPIDValues: Mapper Error"; 142973df0db0SJames Feist messages::internalError(self->asyncResp->res); 143073df0db0SJames Feist return; 143173df0db0SJames Feist } 143273df0db0SJames Feist 143373df0db0SJames Feist const std::string& path = subtree[0].first; 143473df0db0SJames Feist const std::string& owner = subtree[0].second[0].first; 1435fac6e53bSKrzysztof Grobelny sdbusplus::asio::getAllProperties( 1436fac6e53bSKrzysztof Grobelny *crow::connections::systemBus, owner, path, thermalModeIface, 1437002d39b4SEd Tanous [self, path, owner](const boost::system::error_code ec2, 1438b9d36b47SEd Tanous const dbus::utility::DBusPropertiesMap& r) { 1439cb13a392SEd Tanous if (ec2) 144073df0db0SJames Feist { 14410fda0f12SGeorge Liu BMCWEB_LOG_ERROR 1442002d39b4SEd Tanous << "SetPIDValues: Can't get thermalModeIface " << path; 144373df0db0SJames Feist messages::internalError(self->asyncResp->res); 144473df0db0SJames Feist return; 144573df0db0SJames Feist } 1446271584abSEd Tanous const std::string* current = nullptr; 1447271584abSEd Tanous const std::vector<std::string>* supported = nullptr; 1448fac6e53bSKrzysztof Grobelny 1449fac6e53bSKrzysztof Grobelny const bool success = sdbusplus::unpackPropertiesNoThrow( 1450fac6e53bSKrzysztof Grobelny dbus_utils::UnpackErrorPrinter(), r, "Current", current, 1451fac6e53bSKrzysztof Grobelny "Supported", supported); 1452fac6e53bSKrzysztof Grobelny 1453fac6e53bSKrzysztof Grobelny if (!success) 145473df0db0SJames Feist { 1455002d39b4SEd Tanous messages::internalError(self->asyncResp->res); 145673df0db0SJames Feist return; 145773df0db0SJames Feist } 1458fac6e53bSKrzysztof Grobelny 145973df0db0SJames Feist if (current == nullptr || supported == nullptr) 146073df0db0SJames Feist { 14610fda0f12SGeorge Liu BMCWEB_LOG_ERROR 1462002d39b4SEd Tanous << "SetPIDValues: thermal mode iface invalid " << path; 146373df0db0SJames Feist messages::internalError(self->asyncResp->res); 146473df0db0SJames Feist return; 146573df0db0SJames Feist } 146673df0db0SJames Feist self->currentProfile = *current; 146773df0db0SJames Feist self->supportedProfiles = *supported; 146873df0db0SJames Feist self->profileConnection = owner; 146973df0db0SJames Feist self->profilePath = path; 1470fac6e53bSKrzysztof Grobelny }); 14715b4aa86bSJames Feist }, 14725b4aa86bSJames Feist "xyz.openbmc_project.ObjectMapper", 14735b4aa86bSJames Feist "/xyz/openbmc_project/object_mapper", 14745b4aa86bSJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 147573df0db0SJames Feist std::array<const char*, 1>{thermalModeIface}); 147673df0db0SJames Feist } 147724b2fe81SEd Tanous void pidSetDone() 147873df0db0SJames Feist { 147973df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 148073df0db0SJames Feist { 148173df0db0SJames Feist return; 14825b4aa86bSJames Feist } 14838d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> response = asyncResp; 148473df0db0SJames Feist if (profile) 148573df0db0SJames Feist { 148673df0db0SJames Feist if (std::find(supportedProfiles.begin(), supportedProfiles.end(), 148773df0db0SJames Feist *profile) == supportedProfiles.end()) 148873df0db0SJames Feist { 148973df0db0SJames Feist messages::actionParameterUnknown(response->res, "Profile", 149073df0db0SJames Feist *profile); 149173df0db0SJames Feist return; 149273df0db0SJames Feist } 149373df0db0SJames Feist currentProfile = *profile; 149473df0db0SJames Feist crow::connections::systemBus->async_method_call( 149573df0db0SJames Feist [response](const boost::system::error_code ec) { 149673df0db0SJames Feist if (ec) 149773df0db0SJames Feist { 149873df0db0SJames Feist BMCWEB_LOG_ERROR << "Error patching profile" << ec; 149973df0db0SJames Feist messages::internalError(response->res); 150073df0db0SJames Feist } 150173df0db0SJames Feist }, 150273df0db0SJames Feist profileConnection, profilePath, 150373df0db0SJames Feist "org.freedesktop.DBus.Properties", "Set", thermalModeIface, 1504168e20c1SEd Tanous "Current", dbus::utility::DbusVariantType(*profile)); 150573df0db0SJames Feist } 150673df0db0SJames Feist 150773df0db0SJames Feist for (auto& containerPair : configuration) 150873df0db0SJames Feist { 150973df0db0SJames Feist auto& container = containerPair.second; 151073df0db0SJames Feist if (!container) 151173df0db0SJames Feist { 151273df0db0SJames Feist continue; 151373df0db0SJames Feist } 15146ee7f774SJames Feist BMCWEB_LOG_DEBUG << *container; 15156ee7f774SJames Feist 151602cad96eSEd Tanous const std::string& type = containerPair.first; 151773df0db0SJames Feist 151873df0db0SJames Feist for (nlohmann::json::iterator it = container->begin(); 151917a897dfSManojkiran Eda it != container->end(); ++it) 152073df0db0SJames Feist { 152173df0db0SJames Feist const auto& name = it.key(); 15226ee7f774SJames Feist BMCWEB_LOG_DEBUG << "looking for " << name; 15236ee7f774SJames Feist 152473df0db0SJames Feist auto pathItr = 152573df0db0SJames Feist std::find_if(managedObj.begin(), managedObj.end(), 152673df0db0SJames Feist [&name](const auto& obj) { 1527002d39b4SEd Tanous return boost::algorithm::ends_with(obj.first.str, 1528002d39b4SEd Tanous "/" + name); 152973df0db0SJames Feist }); 1530b9d36b47SEd Tanous dbus::utility::DBusPropertiesMap output; 153173df0db0SJames Feist 153273df0db0SJames Feist output.reserve(16); // The pid interface length 153373df0db0SJames Feist 153473df0db0SJames Feist // determines if we're patching entity-manager or 153573df0db0SJames Feist // creating a new object 153673df0db0SJames Feist bool createNewObject = (pathItr == managedObj.end()); 15376ee7f774SJames Feist BMCWEB_LOG_DEBUG << "Found = " << !createNewObject; 15386ee7f774SJames Feist 153973df0db0SJames Feist std::string iface; 1540ea2b670dSEd Tanous if (!createNewObject) 1541ea2b670dSEd Tanous { 15428be2b5b6SPotin Lai bool findInterface = false; 1543ea2b670dSEd Tanous for (const auto& interface : pathItr->second) 1544ea2b670dSEd Tanous { 1545ea2b670dSEd Tanous if (interface.first == pidConfigurationIface) 1546ea2b670dSEd Tanous { 1547ea2b670dSEd Tanous if (type == "PidControllers" || 1548ea2b670dSEd Tanous type == "FanControllers") 154973df0db0SJames Feist { 155073df0db0SJames Feist iface = pidConfigurationIface; 15518be2b5b6SPotin Lai findInterface = true; 15528be2b5b6SPotin Lai break; 155373df0db0SJames Feist } 155473df0db0SJames Feist } 1555ea2b670dSEd Tanous else if (interface.first == pidZoneConfigurationIface) 155673df0db0SJames Feist { 1557ea2b670dSEd Tanous if (type == "FanZones") 155873df0db0SJames Feist { 1559ea2b670dSEd Tanous iface = pidConfigurationIface; 15608be2b5b6SPotin Lai findInterface = true; 15618be2b5b6SPotin Lai break; 156273df0db0SJames Feist } 156373df0db0SJames Feist } 1564ea2b670dSEd Tanous else if (interface.first == stepwiseConfigurationIface) 1565ea2b670dSEd Tanous { 1566ea2b670dSEd Tanous if (type == "StepwiseControllers") 156773df0db0SJames Feist { 156873df0db0SJames Feist iface = stepwiseConfigurationIface; 15698be2b5b6SPotin Lai findInterface = true; 15708be2b5b6SPotin Lai break; 15718be2b5b6SPotin Lai } 15728be2b5b6SPotin Lai } 15738be2b5b6SPotin Lai } 15748be2b5b6SPotin Lai 15758be2b5b6SPotin Lai // create new object if interface not found 15768be2b5b6SPotin Lai if (!findInterface) 15778be2b5b6SPotin Lai { 157873df0db0SJames Feist createNewObject = true; 157973df0db0SJames Feist } 1580ea2b670dSEd Tanous } 15816ee7f774SJames Feist 15826ee7f774SJames Feist if (createNewObject && it.value() == nullptr) 15836ee7f774SJames Feist { 15844e0453b1SGunnar Mills // can't delete a non-existent object 15851668ce6dSEd Tanous messages::propertyValueNotInList(response->res, 15861668ce6dSEd Tanous it.value().dump(), name); 15876ee7f774SJames Feist continue; 15886ee7f774SJames Feist } 15896ee7f774SJames Feist 15906ee7f774SJames Feist std::string path; 15916ee7f774SJames Feist if (pathItr != managedObj.end()) 15926ee7f774SJames Feist { 15936ee7f774SJames Feist path = pathItr->first.str; 15946ee7f774SJames Feist } 15956ee7f774SJames Feist 159673df0db0SJames Feist BMCWEB_LOG_DEBUG << "Create new = " << createNewObject << "\n"; 1597e69d9de2SJames Feist 1598e69d9de2SJames Feist // arbitrary limit to avoid attacks 1599e69d9de2SJames Feist constexpr const size_t controllerLimit = 500; 160014b0b8d5SJames Feist if (createNewObject && objectCount >= controllerLimit) 1601e69d9de2SJames Feist { 1602e69d9de2SJames Feist messages::resourceExhaustion(response->res, type); 1603e69d9de2SJames Feist continue; 1604e69d9de2SJames Feist } 1605a170f275SEd Tanous std::string escaped = name; 1606a170f275SEd Tanous std::replace(escaped.begin(), escaped.end(), '_', ' '); 1607a170f275SEd Tanous output.emplace_back("Name", escaped); 160873df0db0SJames Feist 160973df0db0SJames Feist std::string chassis; 161073df0db0SJames Feist CreatePIDRet ret = createPidInterface( 16116ee7f774SJames Feist response, type, it, path, managedObj, createNewObject, 16126ee7f774SJames Feist output, chassis, currentProfile); 161373df0db0SJames Feist if (ret == CreatePIDRet::fail) 161473df0db0SJames Feist { 161573df0db0SJames Feist return; 161673df0db0SJames Feist } 16173174e4dfSEd Tanous if (ret == CreatePIDRet::del) 161873df0db0SJames Feist { 161973df0db0SJames Feist continue; 162073df0db0SJames Feist } 162173df0db0SJames Feist 162273df0db0SJames Feist if (!createNewObject) 162373df0db0SJames Feist { 162473df0db0SJames Feist for (const auto& property : output) 162573df0db0SJames Feist { 162673df0db0SJames Feist crow::connections::systemBus->async_method_call( 162773df0db0SJames Feist [response, 162873df0db0SJames Feist propertyName{std::string(property.first)}]( 162973df0db0SJames Feist const boost::system::error_code ec) { 163073df0db0SJames Feist if (ec) 163173df0db0SJames Feist { 163273df0db0SJames Feist BMCWEB_LOG_ERROR << "Error patching " 1633002d39b4SEd Tanous << propertyName << ": " << ec; 163473df0db0SJames Feist messages::internalError(response->res); 163573df0db0SJames Feist return; 163673df0db0SJames Feist } 163773df0db0SJames Feist messages::success(response->res); 163873df0db0SJames Feist }, 16396ee7f774SJames Feist "xyz.openbmc_project.EntityManager", path, 164073df0db0SJames Feist "org.freedesktop.DBus.Properties", "Set", iface, 164173df0db0SJames Feist property.first, property.second); 164273df0db0SJames Feist } 164373df0db0SJames Feist } 164473df0db0SJames Feist else 164573df0db0SJames Feist { 164673df0db0SJames Feist if (chassis.empty()) 164773df0db0SJames Feist { 164873df0db0SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config"; 1649ace85d60SEd Tanous messages::internalError(response->res); 165073df0db0SJames Feist return; 165173df0db0SJames Feist } 165273df0db0SJames Feist 165373df0db0SJames Feist bool foundChassis = false; 165473df0db0SJames Feist for (const auto& obj : managedObj) 165573df0db0SJames Feist { 165673df0db0SJames Feist if (boost::algorithm::ends_with(obj.first.str, chassis)) 165773df0db0SJames Feist { 165873df0db0SJames Feist chassis = obj.first.str; 165973df0db0SJames Feist foundChassis = true; 166073df0db0SJames Feist break; 166173df0db0SJames Feist } 166273df0db0SJames Feist } 166373df0db0SJames Feist if (!foundChassis) 166473df0db0SJames Feist { 166573df0db0SJames Feist BMCWEB_LOG_ERROR << "Failed to find chassis on dbus"; 166673df0db0SJames Feist messages::resourceMissingAtURI( 1667ace85d60SEd Tanous response->res, 1668ace85d60SEd Tanous crow::utility::urlFromPieces("redfish", "v1", 1669ace85d60SEd Tanous "Chassis", chassis)); 167073df0db0SJames Feist return; 167173df0db0SJames Feist } 167273df0db0SJames Feist 167373df0db0SJames Feist crow::connections::systemBus->async_method_call( 167473df0db0SJames Feist [response](const boost::system::error_code ec) { 167573df0db0SJames Feist if (ec) 167673df0db0SJames Feist { 167773df0db0SJames Feist BMCWEB_LOG_ERROR << "Error Adding Pid Object " 167873df0db0SJames Feist << ec; 167973df0db0SJames Feist messages::internalError(response->res); 168073df0db0SJames Feist return; 168173df0db0SJames Feist } 168273df0db0SJames Feist messages::success(response->res); 168373df0db0SJames Feist }, 168473df0db0SJames Feist "xyz.openbmc_project.EntityManager", chassis, 168573df0db0SJames Feist "xyz.openbmc_project.AddObject", "AddObject", output); 168673df0db0SJames Feist } 168773df0db0SJames Feist } 168873df0db0SJames Feist } 168973df0db0SJames Feist } 169024b2fe81SEd Tanous 169124b2fe81SEd Tanous ~SetPIDValues() 169224b2fe81SEd Tanous { 169324b2fe81SEd Tanous try 169424b2fe81SEd Tanous { 169524b2fe81SEd Tanous pidSetDone(); 169624b2fe81SEd Tanous } 169724b2fe81SEd Tanous catch (...) 169824b2fe81SEd Tanous { 169924b2fe81SEd Tanous BMCWEB_LOG_CRITICAL << "pidSetDone threw exception"; 170024b2fe81SEd Tanous } 170124b2fe81SEd Tanous } 170224b2fe81SEd Tanous 17038d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> asyncResp; 170473df0db0SJames Feist std::vector<std::pair<std::string, std::optional<nlohmann::json>>> 170573df0db0SJames Feist configuration; 170673df0db0SJames Feist std::optional<std::string> profile; 170773df0db0SJames Feist dbus::utility::ManagedObjectType managedObj; 170873df0db0SJames Feist std::vector<std::string> supportedProfiles; 170973df0db0SJames Feist std::string currentProfile; 171073df0db0SJames Feist std::string profileConnection; 171173df0db0SJames Feist std::string profilePath; 171214b0b8d5SJames Feist size_t objectCount = 0; 171373df0db0SJames Feist }; 171473df0db0SJames Feist 1715071d8fdfSSunnySrivastava1984 /** 1716071d8fdfSSunnySrivastava1984 * @brief Retrieves BMC manager location data over DBus 1717071d8fdfSSunnySrivastava1984 * 1718071d8fdfSSunnySrivastava1984 * @param[in] aResp Shared pointer for completing asynchronous calls 1719071d8fdfSSunnySrivastava1984 * @param[in] connectionName - service name 1720071d8fdfSSunnySrivastava1984 * @param[in] path - object path 1721071d8fdfSSunnySrivastava1984 * @return none 1722071d8fdfSSunnySrivastava1984 */ 17238d1b46d7Szhanghch05 inline void getLocation(const std::shared_ptr<bmcweb::AsyncResp>& aResp, 1724071d8fdfSSunnySrivastava1984 const std::string& connectionName, 1725071d8fdfSSunnySrivastava1984 const std::string& path) 1726071d8fdfSSunnySrivastava1984 { 1727071d8fdfSSunnySrivastava1984 BMCWEB_LOG_DEBUG << "Get BMC manager Location data."; 1728071d8fdfSSunnySrivastava1984 17291e1e598dSJonathan Doman sdbusplus::asio::getProperty<std::string>( 17301e1e598dSJonathan Doman *crow::connections::systemBus, connectionName, path, 17311e1e598dSJonathan Doman "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode", 1732071d8fdfSSunnySrivastava1984 [aResp](const boost::system::error_code ec, 17331e1e598dSJonathan Doman const std::string& property) { 1734071d8fdfSSunnySrivastava1984 if (ec) 1735071d8fdfSSunnySrivastava1984 { 1736071d8fdfSSunnySrivastava1984 BMCWEB_LOG_DEBUG << "DBUS response error for " 1737071d8fdfSSunnySrivastava1984 "Location"; 1738071d8fdfSSunnySrivastava1984 messages::internalError(aResp->res); 1739071d8fdfSSunnySrivastava1984 return; 1740071d8fdfSSunnySrivastava1984 } 1741071d8fdfSSunnySrivastava1984 1742071d8fdfSSunnySrivastava1984 aResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] = 17431e1e598dSJonathan Doman property; 17441e1e598dSJonathan Doman }); 1745071d8fdfSSunnySrivastava1984 } 17467e860f15SJohn Edward Broadbent // avoid name collision systems.hpp 17477e860f15SJohn Edward Broadbent inline void 17487e860f15SJohn Edward Broadbent managerGetLastResetTime(const std::shared_ptr<bmcweb::AsyncResp>& aResp) 17494bf2b033SGunnar Mills { 17504bf2b033SGunnar Mills BMCWEB_LOG_DEBUG << "Getting Manager Last Reset Time"; 17514bf2b033SGunnar Mills 17521e1e598dSJonathan Doman sdbusplus::asio::getProperty<uint64_t>( 17531e1e598dSJonathan Doman *crow::connections::systemBus, "xyz.openbmc_project.State.BMC", 17541e1e598dSJonathan Doman "/xyz/openbmc_project/state/bmc0", "xyz.openbmc_project.State.BMC", 17551e1e598dSJonathan Doman "LastRebootTime", 17564bf2b033SGunnar Mills [aResp](const boost::system::error_code ec, 17571e1e598dSJonathan Doman const uint64_t lastResetTime) { 17584bf2b033SGunnar Mills if (ec) 17594bf2b033SGunnar Mills { 17604bf2b033SGunnar Mills BMCWEB_LOG_DEBUG << "D-BUS response error " << ec; 17614bf2b033SGunnar Mills return; 17624bf2b033SGunnar Mills } 17634bf2b033SGunnar Mills 17644bf2b033SGunnar Mills // LastRebootTime is epoch time, in milliseconds 17654bf2b033SGunnar Mills // https://github.com/openbmc/phosphor-dbus-interfaces/blob/7f9a128eb9296e926422ddc312c148b625890bb6/xyz/openbmc_project/State/BMC.interface.yaml#L19 17661e1e598dSJonathan Doman uint64_t lastResetTimeStamp = lastResetTime / 1000; 17674bf2b033SGunnar Mills 17684bf2b033SGunnar Mills // Convert to ISO 8601 standard 17694bf2b033SGunnar Mills aResp->res.jsonValue["LastResetTime"] = 17702b82937eSEd Tanous redfish::time_utils::getDateTimeUint(lastResetTimeStamp); 17711e1e598dSJonathan Doman }); 17724bf2b033SGunnar Mills } 17734bf2b033SGunnar Mills 17744bfefa74SGunnar Mills /** 17754bfefa74SGunnar Mills * @brief Set the running firmware image 17764bfefa74SGunnar Mills * 17774bfefa74SGunnar Mills * @param[i,o] aResp - Async response object 17784bfefa74SGunnar Mills * @param[i] runningFirmwareTarget - Image to make the running image 17794bfefa74SGunnar Mills * 17804bfefa74SGunnar Mills * @return void 17814bfefa74SGunnar Mills */ 17827e860f15SJohn Edward Broadbent inline void 17837e860f15SJohn Edward Broadbent setActiveFirmwareImage(const std::shared_ptr<bmcweb::AsyncResp>& aResp, 1784f23b7296SEd Tanous const std::string& runningFirmwareTarget) 17854bfefa74SGunnar Mills { 17864bfefa74SGunnar Mills // Get the Id from /redfish/v1/UpdateService/FirmwareInventory/<Id> 1787f23b7296SEd Tanous std::string::size_type idPos = runningFirmwareTarget.rfind('/'); 17884bfefa74SGunnar Mills if (idPos == std::string::npos) 17894bfefa74SGunnar Mills { 17904bfefa74SGunnar Mills messages::propertyValueNotInList(aResp->res, runningFirmwareTarget, 17914bfefa74SGunnar Mills "@odata.id"); 17924bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Can't parse firmware ID!"; 17934bfefa74SGunnar Mills return; 17944bfefa74SGunnar Mills } 17954bfefa74SGunnar Mills idPos++; 17964bfefa74SGunnar Mills if (idPos >= runningFirmwareTarget.size()) 17974bfefa74SGunnar Mills { 17984bfefa74SGunnar Mills messages::propertyValueNotInList(aResp->res, runningFirmwareTarget, 17994bfefa74SGunnar Mills "@odata.id"); 18004bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Invalid firmware ID."; 18014bfefa74SGunnar Mills return; 18024bfefa74SGunnar Mills } 18034bfefa74SGunnar Mills std::string firmwareId = runningFirmwareTarget.substr(idPos); 18044bfefa74SGunnar Mills 18054bfefa74SGunnar Mills // Make sure the image is valid before setting priority 18064bfefa74SGunnar Mills crow::connections::systemBus->async_method_call( 1807711ac7a9SEd Tanous [aResp, firmwareId, 1808711ac7a9SEd Tanous runningFirmwareTarget](const boost::system::error_code ec, 1809711ac7a9SEd Tanous dbus::utility::ManagedObjectType& subtree) { 18104bfefa74SGunnar Mills if (ec) 18114bfefa74SGunnar Mills { 18124bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "D-Bus response error getting objects."; 18134bfefa74SGunnar Mills messages::internalError(aResp->res); 18144bfefa74SGunnar Mills return; 18154bfefa74SGunnar Mills } 18164bfefa74SGunnar Mills 181726f6976fSEd Tanous if (subtree.empty()) 18184bfefa74SGunnar Mills { 18194bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Can't find image!"; 18204bfefa74SGunnar Mills messages::internalError(aResp->res); 18214bfefa74SGunnar Mills return; 18224bfefa74SGunnar Mills } 18234bfefa74SGunnar Mills 18244bfefa74SGunnar Mills bool foundImage = false; 182502cad96eSEd Tanous for (const auto& object : subtree) 18264bfefa74SGunnar Mills { 18274bfefa74SGunnar Mills const std::string& path = 18284bfefa74SGunnar Mills static_cast<const std::string&>(object.first); 1829f23b7296SEd Tanous std::size_t idPos2 = path.rfind('/'); 18304bfefa74SGunnar Mills 18314bfefa74SGunnar Mills if (idPos2 == std::string::npos) 18324bfefa74SGunnar Mills { 18334bfefa74SGunnar Mills continue; 18344bfefa74SGunnar Mills } 18354bfefa74SGunnar Mills 18364bfefa74SGunnar Mills idPos2++; 18374bfefa74SGunnar Mills if (idPos2 >= path.size()) 18384bfefa74SGunnar Mills { 18394bfefa74SGunnar Mills continue; 18404bfefa74SGunnar Mills } 18414bfefa74SGunnar Mills 18424bfefa74SGunnar Mills if (path.substr(idPos2) == firmwareId) 18434bfefa74SGunnar Mills { 18444bfefa74SGunnar Mills foundImage = true; 18454bfefa74SGunnar Mills break; 18464bfefa74SGunnar Mills } 18474bfefa74SGunnar Mills } 18484bfefa74SGunnar Mills 18494bfefa74SGunnar Mills if (!foundImage) 18504bfefa74SGunnar Mills { 1851002d39b4SEd Tanous messages::propertyValueNotInList(aResp->res, runningFirmwareTarget, 1852002d39b4SEd Tanous "@odata.id"); 18534bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Invalid firmware ID."; 18544bfefa74SGunnar Mills return; 18554bfefa74SGunnar Mills } 18564bfefa74SGunnar Mills 18578cc8edecSEd Tanous BMCWEB_LOG_DEBUG << "Setting firmware version " << firmwareId 18588cc8edecSEd Tanous << " to priority 0."; 18594bfefa74SGunnar Mills 18604bfefa74SGunnar Mills // Only support Immediate 18614bfefa74SGunnar Mills // An addition could be a Redfish Setting like 18624bfefa74SGunnar Mills // ActiveSoftwareImageApplyTime and support OnReset 18634bfefa74SGunnar Mills crow::connections::systemBus->async_method_call( 18648a592810SEd Tanous [aResp](const boost::system::error_code ec2) { 18658a592810SEd Tanous if (ec2) 18664bfefa74SGunnar Mills { 18674bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "D-Bus response error setting."; 18684bfefa74SGunnar Mills messages::internalError(aResp->res); 18694bfefa74SGunnar Mills return; 18704bfefa74SGunnar Mills } 18714bfefa74SGunnar Mills doBMCGracefulRestart(aResp); 18724bfefa74SGunnar Mills }, 18734bfefa74SGunnar Mills 18744bfefa74SGunnar Mills "xyz.openbmc_project.Software.BMC.Updater", 18754bfefa74SGunnar Mills "/xyz/openbmc_project/software/" + firmwareId, 18764bfefa74SGunnar Mills "org.freedesktop.DBus.Properties", "Set", 18777e860f15SJohn Edward Broadbent "xyz.openbmc_project.Software.RedundancyPriority", "Priority", 1878168e20c1SEd Tanous dbus::utility::DbusVariantType(static_cast<uint8_t>(0))); 18794bfefa74SGunnar Mills }, 18804bfefa74SGunnar Mills "xyz.openbmc_project.Software.BMC.Updater", 18817e860f15SJohn Edward Broadbent "/xyz/openbmc_project/software", "org.freedesktop.DBus.ObjectManager", 18827e860f15SJohn Edward Broadbent "GetManagedObjects"); 18834bfefa74SGunnar Mills } 18844bfefa74SGunnar Mills 18857e860f15SJohn Edward Broadbent inline void setDateTime(std::shared_ptr<bmcweb::AsyncResp> aResp, 18867e860f15SJohn Edward Broadbent std::string datetime) 1887af5d6058SSantosh Puranik { 1888af5d6058SSantosh Puranik BMCWEB_LOG_DEBUG << "Set date time: " << datetime; 1889af5d6058SSantosh Puranik 1890af5d6058SSantosh Puranik std::stringstream stream(datetime); 1891af5d6058SSantosh Puranik // Convert from ISO 8601 to boost local_time 1892af5d6058SSantosh Puranik // (BMC only has time in UTC) 1893af5d6058SSantosh Puranik boost::posix_time::ptime posixTime; 1894af5d6058SSantosh Puranik boost::posix_time::ptime epoch(boost::gregorian::date(1970, 1, 1)); 1895af5d6058SSantosh Puranik // Facet gets deleted with the stringsteam 1896af5d6058SSantosh Puranik auto ifc = std::make_unique<boost::local_time::local_time_input_facet>( 1897af5d6058SSantosh Puranik "%Y-%m-%d %H:%M:%S%F %ZP"); 1898af5d6058SSantosh Puranik stream.imbue(std::locale(stream.getloc(), ifc.release())); 1899af5d6058SSantosh Puranik 19007e860f15SJohn Edward Broadbent boost::local_time::local_date_time ldt(boost::local_time::not_a_date_time); 1901af5d6058SSantosh Puranik 1902af5d6058SSantosh Puranik if (stream >> ldt) 1903af5d6058SSantosh Puranik { 1904af5d6058SSantosh Puranik posixTime = ldt.utc_time(); 1905af5d6058SSantosh Puranik boost::posix_time::time_duration dur = posixTime - epoch; 19067e860f15SJohn Edward Broadbent uint64_t durMicroSecs = static_cast<uint64_t>(dur.total_microseconds()); 1907af5d6058SSantosh Puranik crow::connections::systemBus->async_method_call( 1908af5d6058SSantosh Puranik [aResp{std::move(aResp)}, datetime{std::move(datetime)}]( 1909af5d6058SSantosh Puranik const boost::system::error_code ec) { 1910af5d6058SSantosh Puranik if (ec) 1911af5d6058SSantosh Puranik { 1912af5d6058SSantosh Puranik BMCWEB_LOG_DEBUG << "Failed to set elapsed time. " 1913af5d6058SSantosh Puranik "DBUS response error " 1914af5d6058SSantosh Puranik << ec; 1915af5d6058SSantosh Puranik messages::internalError(aResp->res); 1916af5d6058SSantosh Puranik return; 1917af5d6058SSantosh Puranik } 1918af5d6058SSantosh Puranik aResp->res.jsonValue["DateTime"] = datetime; 1919af5d6058SSantosh Puranik }, 19207e860f15SJohn Edward Broadbent "xyz.openbmc_project.Time.Manager", "/xyz/openbmc_project/time/bmc", 1921af5d6058SSantosh Puranik "org.freedesktop.DBus.Properties", "Set", 1922af5d6058SSantosh Puranik "xyz.openbmc_project.Time.EpochTime", "Elapsed", 1923168e20c1SEd Tanous dbus::utility::DbusVariantType(durMicroSecs)); 1924af5d6058SSantosh Puranik } 1925af5d6058SSantosh Puranik else 1926af5d6058SSantosh Puranik { 19277e860f15SJohn Edward Broadbent messages::propertyValueFormatError(aResp->res, datetime, "DateTime"); 1928af5d6058SSantosh Puranik return; 1929af5d6058SSantosh Puranik } 193083ff9ab6SJames Feist } 19319c310685SBorawski.Lukasz 19327e860f15SJohn Edward Broadbent inline void requestRoutesManager(App& app) 19337e860f15SJohn Edward Broadbent { 19347e860f15SJohn Edward Broadbent std::string uuid = persistent_data::getConfig().systemUuid; 19359c310685SBorawski.Lukasz 19367e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/") 1937ed398213SEd Tanous .privileges(redfish::privileges::getManager) 1938002d39b4SEd Tanous .methods(boost::beast::http::verb::get)( 1939002d39b4SEd Tanous [&app, uuid](const crow::Request& req, 194045ca1b86SEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 19413ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 194245ca1b86SEd Tanous { 194345ca1b86SEd Tanous return; 194445ca1b86SEd Tanous } 19457e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc"; 1946a51fc2d2SSui Chen asyncResp->res.jsonValue["@odata.type"] = "#Manager.v1_14_0.Manager"; 19477e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Id"] = "bmc"; 19487e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Name"] = "OpenBmc Manager"; 19497e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Description"] = 19507e860f15SJohn Edward Broadbent "Baseboard Management Controller"; 19517e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["PowerState"] = "On"; 19521476687dSEd Tanous asyncResp->res.jsonValue["Status"]["State"] = "Enabled"; 19531476687dSEd Tanous asyncResp->res.jsonValue["Status"]["Health"] = "OK"; 19541476687dSEd Tanous 19557e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["ManagerType"] = "BMC"; 19567e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["UUID"] = systemd_utils::getUuid(); 19577e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["ServiceEntryPointUUID"] = uuid; 1958002d39b4SEd Tanous asyncResp->res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model 19597e860f15SJohn Edward Broadbent 19601476687dSEd Tanous asyncResp->res.jsonValue["LogServices"]["@odata.id"] = 19611476687dSEd Tanous "/redfish/v1/Managers/bmc/LogServices"; 19621476687dSEd Tanous asyncResp->res.jsonValue["NetworkProtocol"]["@odata.id"] = 19631476687dSEd Tanous "/redfish/v1/Managers/bmc/NetworkProtocol"; 19641476687dSEd Tanous asyncResp->res.jsonValue["EthernetInterfaces"]["@odata.id"] = 19651476687dSEd Tanous "/redfish/v1/Managers/bmc/EthernetInterfaces"; 19667e860f15SJohn Edward Broadbent 19677e860f15SJohn Edward Broadbent #ifdef BMCWEB_ENABLE_VM_NBDPROXY 19681476687dSEd Tanous asyncResp->res.jsonValue["VirtualMedia"]["@odata.id"] = 19691476687dSEd Tanous "/redfish/v1/Managers/bmc/VirtualMedia"; 19707e860f15SJohn Edward Broadbent #endif // BMCWEB_ENABLE_VM_NBDPROXY 19717e860f15SJohn Edward Broadbent 19727e860f15SJohn Edward Broadbent // default oem data 19737e860f15SJohn Edward Broadbent nlohmann::json& oem = asyncResp->res.jsonValue["Oem"]; 19747e860f15SJohn Edward Broadbent nlohmann::json& oemOpenbmc = oem["OpenBmc"]; 19757e860f15SJohn Edward Broadbent oem["@odata.type"] = "#OemManager.Oem"; 19767e860f15SJohn Edward Broadbent oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem"; 19777e860f15SJohn Edward Broadbent oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc"; 19787e860f15SJohn Edward Broadbent oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc"; 19791476687dSEd Tanous 19801476687dSEd Tanous nlohmann::json::object_t certificates; 19811476687dSEd Tanous certificates["@odata.id"] = 19821476687dSEd Tanous "/redfish/v1/Managers/bmc/Truststore/Certificates"; 19831476687dSEd Tanous oemOpenbmc["Certificates"] = std::move(certificates); 19847e860f15SJohn Edward Broadbent 19857e860f15SJohn Edward Broadbent // Manager.Reset (an action) can be many values, OpenBMC only 19867e860f15SJohn Edward Broadbent // supports BMC reboot. 19877e860f15SJohn Edward Broadbent nlohmann::json& managerReset = 19887e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Actions"]["#Manager.Reset"]; 19897e860f15SJohn Edward Broadbent managerReset["target"] = 19907e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/Actions/Manager.Reset"; 19917e860f15SJohn Edward Broadbent managerReset["@Redfish.ActionInfo"] = 19927e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/ResetActionInfo"; 19937e860f15SJohn Edward Broadbent 19947e860f15SJohn Edward Broadbent // ResetToDefaults (Factory Reset) has values like 19957e860f15SJohn Edward Broadbent // PreserveNetworkAndUsers and PreserveNetwork that aren't supported 19967e860f15SJohn Edward Broadbent // on OpenBMC 19977e860f15SJohn Edward Broadbent nlohmann::json& resetToDefaults = 19987e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Actions"]["#Manager.ResetToDefaults"]; 19997e860f15SJohn Edward Broadbent resetToDefaults["target"] = 20007e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults"; 2001613dabeaSEd Tanous resetToDefaults["ResetType@Redfish.AllowableValues"] = 2002613dabeaSEd Tanous nlohmann::json::array_t({"ResetAll"}); 20037e860f15SJohn Edward Broadbent 20047c8c4058STejas Patil std::pair<std::string, std::string> redfishDateTimeOffset = 20052b82937eSEd Tanous redfish::time_utils::getDateTimeOffsetNow(); 20067c8c4058STejas Patil 20077c8c4058STejas Patil asyncResp->res.jsonValue["DateTime"] = redfishDateTimeOffset.first; 20087c8c4058STejas Patil asyncResp->res.jsonValue["DateTimeLocalOffset"] = 20097c8c4058STejas Patil redfishDateTimeOffset.second; 20107e860f15SJohn Edward Broadbent 20110e8ac5e7SGunnar Mills // TODO (Gunnar): Remove these one day since moved to ComputerSystem 20120e8ac5e7SGunnar Mills // Still used by OCP profiles 20130e8ac5e7SGunnar Mills // https://github.com/opencomputeproject/OCP-Profiles/issues/23 20147e860f15SJohn Edward Broadbent // Fill in SerialConsole info 20157e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["SerialConsole"]["ServiceEnabled"] = true; 2016002d39b4SEd Tanous asyncResp->res.jsonValue["SerialConsole"]["MaxConcurrentSessions"] = 15; 2017613dabeaSEd Tanous asyncResp->res.jsonValue["SerialConsole"]["ConnectTypesSupported"] = 2018613dabeaSEd Tanous nlohmann::json::array_t({"IPMI", "SSH"}); 20197e860f15SJohn Edward Broadbent #ifdef BMCWEB_ENABLE_KVM 20207e860f15SJohn Edward Broadbent // Fill in GraphicalConsole info 2021002d39b4SEd Tanous asyncResp->res.jsonValue["GraphicalConsole"]["ServiceEnabled"] = true; 2022002d39b4SEd Tanous asyncResp->res.jsonValue["GraphicalConsole"]["MaxConcurrentSessions"] = 2023002d39b4SEd Tanous 4; 2024613dabeaSEd Tanous asyncResp->res.jsonValue["GraphicalConsole"]["ConnectTypesSupported"] = 2025613dabeaSEd Tanous nlohmann::json::array_t({"KVMIP"}); 20267e860f15SJohn Edward Broadbent #endif // BMCWEB_ENABLE_KVM 20277e860f15SJohn Edward Broadbent 2028002d39b4SEd Tanous asyncResp->res.jsonValue["Links"]["ManagerForServers@odata.count"] = 1; 20291476687dSEd Tanous 20301476687dSEd Tanous nlohmann::json::array_t managerForServers; 20311476687dSEd Tanous nlohmann::json::object_t manager; 20321476687dSEd Tanous manager["@odata.id"] = "/redfish/v1/Systems/system"; 20331476687dSEd Tanous managerForServers.push_back(std::move(manager)); 20341476687dSEd Tanous 20351476687dSEd Tanous asyncResp->res.jsonValue["Links"]["ManagerForServers"] = 20361476687dSEd Tanous std::move(managerForServers); 20377e860f15SJohn Edward Broadbent 20387e860f15SJohn Edward Broadbent auto health = std::make_shared<HealthPopulate>(asyncResp); 20397e860f15SJohn Edward Broadbent health->isManagersHealth = true; 20407e860f15SJohn Edward Broadbent health->populate(); 20417e860f15SJohn Edward Broadbent 2042eee0013eSWilly Tu sw_util::populateSoftwareInformation(asyncResp, sw_util::bmcPurpose, 20437e860f15SJohn Edward Broadbent "FirmwareVersion", true); 20447e860f15SJohn Edward Broadbent 20457e860f15SJohn Edward Broadbent managerGetLastResetTime(asyncResp); 20467e860f15SJohn Edward Broadbent 2047a51fc2d2SSui Chen // ManagerDiagnosticData is added for all BMCs. 2048a51fc2d2SSui Chen nlohmann::json& managerDiagnosticData = 2049a51fc2d2SSui Chen asyncResp->res.jsonValue["ManagerDiagnosticData"]; 2050a51fc2d2SSui Chen managerDiagnosticData["@odata.id"] = 2051a51fc2d2SSui Chen "/redfish/v1/Managers/bmc/ManagerDiagnosticData"; 2052a51fc2d2SSui Chen 205354dce7f5SGunnar Mills #ifdef BMCWEB_ENABLE_REDFISH_OEM_MANAGER_FAN_DATA 20547e860f15SJohn Edward Broadbent auto pids = std::make_shared<GetPIDValues>(asyncResp); 20557e860f15SJohn Edward Broadbent pids->run(); 205654dce7f5SGunnar Mills #endif 20577e860f15SJohn Edward Broadbent 2058002d39b4SEd Tanous getMainChassisId(asyncResp, 2059002d39b4SEd Tanous [](const std::string& chassisId, 2060002d39b4SEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& aRsp) { 2061002d39b4SEd Tanous aRsp->res.jsonValue["Links"]["ManagerForChassis@odata.count"] = 1; 20621476687dSEd Tanous nlohmann::json::array_t managerForChassis; 20638a592810SEd Tanous nlohmann::json::object_t managerObj; 20648a592810SEd Tanous managerObj["@odata.id"] = "/redfish/v1/Chassis/" + chassisId; 20658a592810SEd Tanous managerForChassis.push_back(std::move(managerObj)); 20661476687dSEd Tanous aRsp->res.jsonValue["Links"]["ManagerForChassis"] = 20671476687dSEd Tanous std::move(managerForChassis); 20681476687dSEd Tanous aRsp->res.jsonValue["Links"]["ManagerInChassis"]["@odata.id"] = 20691476687dSEd Tanous "/redfish/v1/Chassis/" + chassisId; 20707e860f15SJohn Edward Broadbent }); 20717e860f15SJohn Edward Broadbent 20727e860f15SJohn Edward Broadbent static bool started = false; 20737e860f15SJohn Edward Broadbent 20747e860f15SJohn Edward Broadbent if (!started) 20751abe55efSEd Tanous { 20761e1e598dSJonathan Doman sdbusplus::asio::getProperty<double>( 20771e1e598dSJonathan Doman *crow::connections::systemBus, "org.freedesktop.systemd1", 2078002d39b4SEd Tanous "/org/freedesktop/systemd1", "org.freedesktop.systemd1.Manager", 2079002d39b4SEd Tanous "Progress", 20807e860f15SJohn Edward Broadbent [asyncResp](const boost::system::error_code ec, 20811e1e598dSJonathan Doman const double& val) { 20827e860f15SJohn Edward Broadbent if (ec) 20831abe55efSEd Tanous { 20847e860f15SJohn Edward Broadbent BMCWEB_LOG_ERROR << "Error while getting progress"; 20857e860f15SJohn Edward Broadbent messages::internalError(asyncResp->res); 20867e860f15SJohn Edward Broadbent return; 20877e860f15SJohn Edward Broadbent } 20881e1e598dSJonathan Doman if (val < 1.0) 20897e860f15SJohn Edward Broadbent { 2090002d39b4SEd Tanous asyncResp->res.jsonValue["Status"]["State"] = "Starting"; 20917e860f15SJohn Edward Broadbent started = true; 20927e860f15SJohn Edward Broadbent } 20931e1e598dSJonathan Doman }); 20949c310685SBorawski.Lukasz } 20959c310685SBorawski.Lukasz 20967e860f15SJohn Edward Broadbent crow::connections::systemBus->async_method_call( 20977e860f15SJohn Edward Broadbent [asyncResp]( 20987e860f15SJohn Edward Broadbent const boost::system::error_code ec, 2099b9d36b47SEd Tanous const dbus::utility::MapperGetSubTreeResponse& subtree) { 21007e860f15SJohn Edward Broadbent if (ec) 21011abe55efSEd Tanous { 2102002d39b4SEd Tanous BMCWEB_LOG_DEBUG << "D-Bus response error on GetSubTree " << ec; 21037e860f15SJohn Edward Broadbent return; 21047e860f15SJohn Edward Broadbent } 210526f6976fSEd Tanous if (subtree.empty()) 21067e860f15SJohn Edward Broadbent { 21077e860f15SJohn Edward Broadbent BMCWEB_LOG_DEBUG << "Can't find bmc D-Bus object!"; 21087e860f15SJohn Edward Broadbent return; 21097e860f15SJohn Edward Broadbent } 21107e860f15SJohn Edward Broadbent // Assume only 1 bmc D-Bus object 21117e860f15SJohn Edward Broadbent // Throw an error if there is more than 1 21127e860f15SJohn Edward Broadbent if (subtree.size() > 1) 21137e860f15SJohn Edward Broadbent { 2114002d39b4SEd Tanous BMCWEB_LOG_DEBUG << "Found more than 1 bmc D-Bus object!"; 21157e860f15SJohn Edward Broadbent messages::internalError(asyncResp->res); 21167e860f15SJohn Edward Broadbent return; 21177e860f15SJohn Edward Broadbent } 21187e860f15SJohn Edward Broadbent 2119002d39b4SEd Tanous if (subtree[0].first.empty() || subtree[0].second.size() != 1) 21207e860f15SJohn Edward Broadbent { 21217e860f15SJohn Edward Broadbent BMCWEB_LOG_DEBUG << "Error getting bmc D-Bus object!"; 21227e860f15SJohn Edward Broadbent messages::internalError(asyncResp->res); 21237e860f15SJohn Edward Broadbent return; 21247e860f15SJohn Edward Broadbent } 21257e860f15SJohn Edward Broadbent 21267e860f15SJohn Edward Broadbent const std::string& path = subtree[0].first; 2127002d39b4SEd Tanous const std::string& connectionName = subtree[0].second[0].first; 21287e860f15SJohn Edward Broadbent 2129002d39b4SEd Tanous for (const auto& interfaceName : subtree[0].second[0].second) 21307e860f15SJohn Edward Broadbent { 21317e860f15SJohn Edward Broadbent if (interfaceName == 21327e860f15SJohn Edward Broadbent "xyz.openbmc_project.Inventory.Decorator.Asset") 21337e860f15SJohn Edward Broadbent { 2134fac6e53bSKrzysztof Grobelny 2135fac6e53bSKrzysztof Grobelny sdbusplus::asio::getAllProperties( 2136fac6e53bSKrzysztof Grobelny *crow::connections::systemBus, connectionName, path, 2137fac6e53bSKrzysztof Grobelny "xyz.openbmc_project.Inventory.Decorator.Asset", 21388a592810SEd Tanous [asyncResp](const boost::system::error_code ec2, 2139b9d36b47SEd Tanous const dbus::utility::DBusPropertiesMap& 21407e860f15SJohn Edward Broadbent propertiesList) { 21418a592810SEd Tanous if (ec2) 21427e860f15SJohn Edward Broadbent { 2143002d39b4SEd Tanous BMCWEB_LOG_DEBUG << "Can't get bmc asset!"; 21447e860f15SJohn Edward Broadbent return; 21457e860f15SJohn Edward Broadbent } 21467e860f15SJohn Edward Broadbent 2147fac6e53bSKrzysztof Grobelny const std::string* partNumber = nullptr; 2148fac6e53bSKrzysztof Grobelny const std::string* serialNumber = nullptr; 2149fac6e53bSKrzysztof Grobelny const std::string* manufacturer = nullptr; 2150fac6e53bSKrzysztof Grobelny const std::string* model = nullptr; 2151fac6e53bSKrzysztof Grobelny const std::string* sparePartNumber = nullptr; 2152fac6e53bSKrzysztof Grobelny 2153fac6e53bSKrzysztof Grobelny const bool success = sdbusplus::unpackPropertiesNoThrow( 2154fac6e53bSKrzysztof Grobelny dbus_utils::UnpackErrorPrinter(), propertiesList, 2155fac6e53bSKrzysztof Grobelny "PartNumber", partNumber, "SerialNumber", 2156fac6e53bSKrzysztof Grobelny serialNumber, "Manufacturer", manufacturer, "Model", 2157fac6e53bSKrzysztof Grobelny model, "SparePartNumber", sparePartNumber); 2158fac6e53bSKrzysztof Grobelny 2159fac6e53bSKrzysztof Grobelny if (!success) 21607e860f15SJohn Edward Broadbent { 2161002d39b4SEd Tanous messages::internalError(asyncResp->res); 21627e860f15SJohn Edward Broadbent return; 21637e860f15SJohn Edward Broadbent } 2164fac6e53bSKrzysztof Grobelny 2165fac6e53bSKrzysztof Grobelny if (partNumber != nullptr) 2166fac6e53bSKrzysztof Grobelny { 2167fac6e53bSKrzysztof Grobelny asyncResp->res.jsonValue["PartNumber"] = 2168fac6e53bSKrzysztof Grobelny *partNumber; 21697e860f15SJohn Edward Broadbent } 2170fac6e53bSKrzysztof Grobelny 2171fac6e53bSKrzysztof Grobelny if (serialNumber != nullptr) 2172fac6e53bSKrzysztof Grobelny { 2173fac6e53bSKrzysztof Grobelny asyncResp->res.jsonValue["SerialNumber"] = 2174fac6e53bSKrzysztof Grobelny *serialNumber; 21757e860f15SJohn Edward Broadbent } 2176fac6e53bSKrzysztof Grobelny 2177fac6e53bSKrzysztof Grobelny if (manufacturer != nullptr) 2178fac6e53bSKrzysztof Grobelny { 2179fac6e53bSKrzysztof Grobelny asyncResp->res.jsonValue["Manufacturer"] = 2180fac6e53bSKrzysztof Grobelny *manufacturer; 2181fac6e53bSKrzysztof Grobelny } 2182fac6e53bSKrzysztof Grobelny 2183fac6e53bSKrzysztof Grobelny if (model != nullptr) 2184fac6e53bSKrzysztof Grobelny { 2185fac6e53bSKrzysztof Grobelny asyncResp->res.jsonValue["Model"] = *model; 2186fac6e53bSKrzysztof Grobelny } 2187fac6e53bSKrzysztof Grobelny 2188fac6e53bSKrzysztof Grobelny if (sparePartNumber != nullptr) 2189fac6e53bSKrzysztof Grobelny { 2190fac6e53bSKrzysztof Grobelny asyncResp->res.jsonValue["SparePartNumber"] = 2191fac6e53bSKrzysztof Grobelny *sparePartNumber; 2192fac6e53bSKrzysztof Grobelny } 2193fac6e53bSKrzysztof Grobelny }); 21947e860f15SJohn Edward Broadbent } 2195002d39b4SEd Tanous else if (interfaceName == 21960fda0f12SGeorge Liu "xyz.openbmc_project.Inventory.Decorator.LocationCode") 21977e860f15SJohn Edward Broadbent { 21987e860f15SJohn Edward Broadbent getLocation(asyncResp, connectionName, path); 21997e860f15SJohn Edward Broadbent } 22007e860f15SJohn Edward Broadbent } 22017e860f15SJohn Edward Broadbent }, 22027e860f15SJohn Edward Broadbent "xyz.openbmc_project.ObjectMapper", 22037e860f15SJohn Edward Broadbent "/xyz/openbmc_project/object_mapper", 22047e860f15SJohn Edward Broadbent "xyz.openbmc_project.ObjectMapper", "GetSubTree", 22057e860f15SJohn Edward Broadbent "/xyz/openbmc_project/inventory", int32_t(0), 22067e860f15SJohn Edward Broadbent std::array<const char*, 1>{ 22077e860f15SJohn Edward Broadbent "xyz.openbmc_project.Inventory.Item.Bmc"}); 22087e860f15SJohn Edward Broadbent }); 22097e860f15SJohn Edward Broadbent 22107e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/") 2211ed398213SEd Tanous .privileges(redfish::privileges::patchManager) 221245ca1b86SEd Tanous .methods(boost::beast::http::verb::patch)( 221345ca1b86SEd Tanous [&app](const crow::Request& req, 22147e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 22153ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 221645ca1b86SEd Tanous { 221745ca1b86SEd Tanous return; 221845ca1b86SEd Tanous } 22197e860f15SJohn Edward Broadbent std::optional<nlohmann::json> oem; 22207e860f15SJohn Edward Broadbent std::optional<nlohmann::json> links; 22217e860f15SJohn Edward Broadbent std::optional<std::string> datetime; 22227e860f15SJohn Edward Broadbent 222315ed6780SWilly Tu if (!json_util::readJsonPatch(req, asyncResp->res, "Oem", oem, 2224002d39b4SEd Tanous "DateTime", datetime, "Links", links)) 22257e860f15SJohn Edward Broadbent { 22267e860f15SJohn Edward Broadbent return; 22277e860f15SJohn Edward Broadbent } 22287e860f15SJohn Edward Broadbent 22297e860f15SJohn Edward Broadbent if (oem) 22307e860f15SJohn Edward Broadbent { 223154dce7f5SGunnar Mills #ifdef BMCWEB_ENABLE_REDFISH_OEM_MANAGER_FAN_DATA 22327e860f15SJohn Edward Broadbent std::optional<nlohmann::json> openbmc; 2233002d39b4SEd Tanous if (!redfish::json_util::readJson(*oem, asyncResp->res, "OpenBmc", 2234002d39b4SEd Tanous openbmc)) 22357e860f15SJohn Edward Broadbent { 22367e860f15SJohn Edward Broadbent BMCWEB_LOG_ERROR 22377e860f15SJohn Edward Broadbent << "Illegal Property " 2238002d39b4SEd Tanous << oem->dump(2, ' ', true, 22397e860f15SJohn Edward Broadbent nlohmann::json::error_handler_t::replace); 22407e860f15SJohn Edward Broadbent return; 22417e860f15SJohn Edward Broadbent } 22427e860f15SJohn Edward Broadbent if (openbmc) 22437e860f15SJohn Edward Broadbent { 22447e860f15SJohn Edward Broadbent std::optional<nlohmann::json> fan; 2245002d39b4SEd Tanous if (!redfish::json_util::readJson(*openbmc, asyncResp->res, 2246002d39b4SEd Tanous "Fan", fan)) 22477e860f15SJohn Edward Broadbent { 22487e860f15SJohn Edward Broadbent BMCWEB_LOG_ERROR 22497e860f15SJohn Edward Broadbent << "Illegal Property " 2250002d39b4SEd Tanous << openbmc->dump( 2251002d39b4SEd Tanous 2, ' ', true, 2252002d39b4SEd Tanous nlohmann::json::error_handler_t::replace); 22537e860f15SJohn Edward Broadbent return; 22547e860f15SJohn Edward Broadbent } 22557e860f15SJohn Edward Broadbent if (fan) 22567e860f15SJohn Edward Broadbent { 2257002d39b4SEd Tanous auto pid = std::make_shared<SetPIDValues>(asyncResp, *fan); 22587e860f15SJohn Edward Broadbent pid->run(); 22597e860f15SJohn Edward Broadbent } 22607e860f15SJohn Edward Broadbent } 226154dce7f5SGunnar Mills #else 226254dce7f5SGunnar Mills messages::propertyUnknown(asyncResp->res, "Oem"); 226354dce7f5SGunnar Mills return; 226454dce7f5SGunnar Mills #endif 22657e860f15SJohn Edward Broadbent } 22667e860f15SJohn Edward Broadbent if (links) 22677e860f15SJohn Edward Broadbent { 22687e860f15SJohn Edward Broadbent std::optional<nlohmann::json> activeSoftwareImage; 22697e860f15SJohn Edward Broadbent if (!redfish::json_util::readJson(*links, asyncResp->res, 22707e860f15SJohn Edward Broadbent "ActiveSoftwareImage", 22717e860f15SJohn Edward Broadbent activeSoftwareImage)) 22727e860f15SJohn Edward Broadbent { 22737e860f15SJohn Edward Broadbent return; 22747e860f15SJohn Edward Broadbent } 22757e860f15SJohn Edward Broadbent if (activeSoftwareImage) 22767e860f15SJohn Edward Broadbent { 22777e860f15SJohn Edward Broadbent std::optional<std::string> odataId; 2278002d39b4SEd Tanous if (!json_util::readJson(*activeSoftwareImage, asyncResp->res, 2279002d39b4SEd Tanous "@odata.id", odataId)) 22807e860f15SJohn Edward Broadbent { 22817e860f15SJohn Edward Broadbent return; 22827e860f15SJohn Edward Broadbent } 22837e860f15SJohn Edward Broadbent 22847e860f15SJohn Edward Broadbent if (odataId) 22857e860f15SJohn Edward Broadbent { 22867e860f15SJohn Edward Broadbent setActiveFirmwareImage(asyncResp, *odataId); 22877e860f15SJohn Edward Broadbent } 22887e860f15SJohn Edward Broadbent } 22897e860f15SJohn Edward Broadbent } 22907e860f15SJohn Edward Broadbent if (datetime) 22917e860f15SJohn Edward Broadbent { 22927e860f15SJohn Edward Broadbent setDateTime(asyncResp, std::move(*datetime)); 22937e860f15SJohn Edward Broadbent } 22947e860f15SJohn Edward Broadbent }); 22957e860f15SJohn Edward Broadbent } 22967e860f15SJohn Edward Broadbent 22977e860f15SJohn Edward Broadbent inline void requestRoutesManagerCollection(App& app) 22987e860f15SJohn Edward Broadbent { 22997e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/") 2300ed398213SEd Tanous .privileges(redfish::privileges::getManagerCollection) 23017e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)( 230245ca1b86SEd Tanous [&app](const crow::Request& req, 23037e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 23043ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 230545ca1b86SEd Tanous { 230645ca1b86SEd Tanous return; 230745ca1b86SEd Tanous } 230883ff9ab6SJames Feist // Collections don't include the static data added by SubRoute 230983ff9ab6SJames Feist // because it has a duplicate entry for members 23108d1b46d7Szhanghch05 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers"; 23118d1b46d7Szhanghch05 asyncResp->res.jsonValue["@odata.type"] = 23128d1b46d7Szhanghch05 "#ManagerCollection.ManagerCollection"; 23138d1b46d7Szhanghch05 asyncResp->res.jsonValue["Name"] = "Manager Collection"; 23148d1b46d7Szhanghch05 asyncResp->res.jsonValue["Members@odata.count"] = 1; 23151476687dSEd Tanous nlohmann::json::array_t members; 23161476687dSEd Tanous nlohmann::json& bmc = members.emplace_back(); 23171476687dSEd Tanous bmc["@odata.id"] = "/redfish/v1/Managers/bmc"; 23181476687dSEd Tanous asyncResp->res.jsonValue["Members"] = std::move(members); 23197e860f15SJohn Edward Broadbent }); 23209c310685SBorawski.Lukasz } 23219c310685SBorawski.Lukasz } // namespace redfish 2322