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 18b49ac873SJames Feist #include "health.hpp" 19c5d03ff4SJennifer Lee #include "redfish_util.hpp" 209c310685SBorawski.Lukasz 217e860f15SJohn Edward Broadbent #include <app.hpp> 225b4aa86bSJames Feist #include <boost/algorithm/string/replace.hpp> 23af5d6058SSantosh Puranik #include <boost/date_time.hpp> 245b4aa86bSJames Feist #include <dbus_utility.hpp> 25ed398213SEd Tanous #include <registries/privilege_registry.hpp> 26e90c5052SAndrew Geissler #include <utils/fw_utils.hpp> 277bffdb7eSBernard Wong #include <utils/systemd_utils.hpp> 281214b7e7SGunnar Mills 294bfefa74SGunnar Mills #include <cstdint> 301214b7e7SGunnar Mills #include <memory> 311214b7e7SGunnar Mills #include <sstream> 32abf2add6SEd Tanous #include <variant> 335b4aa86bSJames Feist 341abe55efSEd Tanous namespace redfish 351abe55efSEd Tanous { 36ed5befbdSJennifer Lee 37ed5befbdSJennifer Lee /** 382a5c4407SGunnar Mills * Function reboots the BMC. 392a5c4407SGunnar Mills * 402a5c4407SGunnar Mills * @param[in] asyncResp - Shared pointer for completing asynchronous calls 41ed5befbdSJennifer Lee */ 428d1b46d7Szhanghch05 inline void 438d1b46d7Szhanghch05 doBMCGracefulRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 44ed5befbdSJennifer Lee { 45ed5befbdSJennifer Lee const char* processName = "xyz.openbmc_project.State.BMC"; 46ed5befbdSJennifer Lee const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 47ed5befbdSJennifer Lee const char* interfaceName = "xyz.openbmc_project.State.BMC"; 48ed5befbdSJennifer Lee const std::string& propertyValue = 49ed5befbdSJennifer Lee "xyz.openbmc_project.State.BMC.Transition.Reboot"; 50ed5befbdSJennifer Lee const char* destProperty = "RequestedBMCTransition"; 51ed5befbdSJennifer Lee 52ed5befbdSJennifer Lee // Create the D-Bus variant for D-Bus call. 53168e20c1SEd Tanous dbus::utility::DbusVariantType dbusPropertyValue(propertyValue); 54ed5befbdSJennifer Lee 55ed5befbdSJennifer Lee crow::connections::systemBus->async_method_call( 56ed5befbdSJennifer Lee [asyncResp](const boost::system::error_code ec) { 57ed5befbdSJennifer Lee // Use "Set" method to set the property value. 58ed5befbdSJennifer Lee if (ec) 59ed5befbdSJennifer Lee { 602a5c4407SGunnar Mills BMCWEB_LOG_DEBUG << "[Set] Bad D-Bus request error: " << ec; 61ed5befbdSJennifer Lee messages::internalError(asyncResp->res); 62ed5befbdSJennifer Lee return; 63ed5befbdSJennifer Lee } 64ed5befbdSJennifer Lee 65ed5befbdSJennifer Lee messages::success(asyncResp->res); 66ed5befbdSJennifer Lee }, 67ed5befbdSJennifer Lee processName, objectPath, "org.freedesktop.DBus.Properties", "Set", 68ed5befbdSJennifer Lee interfaceName, destProperty, dbusPropertyValue); 69ed5befbdSJennifer Lee } 702a5c4407SGunnar Mills 718d1b46d7Szhanghch05 inline void 728d1b46d7Szhanghch05 doBMCForceRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 73f92af389SJayaprakash Mutyala { 74f92af389SJayaprakash Mutyala const char* processName = "xyz.openbmc_project.State.BMC"; 75f92af389SJayaprakash Mutyala const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 76f92af389SJayaprakash Mutyala const char* interfaceName = "xyz.openbmc_project.State.BMC"; 77f92af389SJayaprakash Mutyala const std::string& propertyValue = 78f92af389SJayaprakash Mutyala "xyz.openbmc_project.State.BMC.Transition.HardReboot"; 79f92af389SJayaprakash Mutyala const char* destProperty = "RequestedBMCTransition"; 80f92af389SJayaprakash Mutyala 81f92af389SJayaprakash Mutyala // Create the D-Bus variant for D-Bus call. 82168e20c1SEd Tanous dbus::utility::DbusVariantType dbusPropertyValue(propertyValue); 83f92af389SJayaprakash Mutyala 84f92af389SJayaprakash Mutyala crow::connections::systemBus->async_method_call( 85f92af389SJayaprakash Mutyala [asyncResp](const boost::system::error_code ec) { 86f92af389SJayaprakash Mutyala // Use "Set" method to set the property value. 87f92af389SJayaprakash Mutyala if (ec) 88f92af389SJayaprakash Mutyala { 89f92af389SJayaprakash Mutyala BMCWEB_LOG_DEBUG << "[Set] Bad D-Bus request error: " << ec; 90f92af389SJayaprakash Mutyala messages::internalError(asyncResp->res); 91f92af389SJayaprakash Mutyala return; 92f92af389SJayaprakash Mutyala } 93f92af389SJayaprakash Mutyala 94f92af389SJayaprakash Mutyala messages::success(asyncResp->res); 95f92af389SJayaprakash Mutyala }, 96f92af389SJayaprakash Mutyala processName, objectPath, "org.freedesktop.DBus.Properties", "Set", 97f92af389SJayaprakash Mutyala interfaceName, destProperty, dbusPropertyValue); 98f92af389SJayaprakash Mutyala } 99f92af389SJayaprakash Mutyala 1002a5c4407SGunnar Mills /** 1012a5c4407SGunnar Mills * ManagerResetAction class supports the POST method for the Reset (reboot) 1022a5c4407SGunnar Mills * action. 1032a5c4407SGunnar Mills */ 1047e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetAction(App& app) 1052a5c4407SGunnar Mills { 1062a5c4407SGunnar Mills /** 1072a5c4407SGunnar Mills * Function handles POST method request. 1082a5c4407SGunnar Mills * Analyzes POST body before sending Reset (Reboot) request data to D-Bus. 109f92af389SJayaprakash Mutyala * OpenBMC supports ResetType "GracefulRestart" and "ForceRestart". 1102a5c4407SGunnar Mills */ 1117e860f15SJohn Edward Broadbent 1127e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/") 113ed398213SEd Tanous .privileges(redfish::privileges::postManager) 1147e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::post)( 1157e860f15SJohn Edward Broadbent [](const crow::Request& req, 1167e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 1172a5c4407SGunnar Mills BMCWEB_LOG_DEBUG << "Post Manager Reset."; 1182a5c4407SGunnar Mills 1192a5c4407SGunnar Mills std::string resetType; 1202a5c4407SGunnar Mills 12115ed6780SWilly Tu if (!json_util::readJsonAction(req, asyncResp->res, "ResetType", 1227e860f15SJohn Edward Broadbent resetType)) 1232a5c4407SGunnar Mills { 1242a5c4407SGunnar Mills return; 1252a5c4407SGunnar Mills } 1262a5c4407SGunnar Mills 127f92af389SJayaprakash Mutyala if (resetType == "GracefulRestart") 128f92af389SJayaprakash Mutyala { 129f92af389SJayaprakash Mutyala BMCWEB_LOG_DEBUG << "Proceeding with " << resetType; 130f92af389SJayaprakash Mutyala doBMCGracefulRestart(asyncResp); 131f92af389SJayaprakash Mutyala return; 132f92af389SJayaprakash Mutyala } 1333174e4dfSEd Tanous if (resetType == "ForceRestart") 134f92af389SJayaprakash Mutyala { 135f92af389SJayaprakash Mutyala BMCWEB_LOG_DEBUG << "Proceeding with " << resetType; 136f92af389SJayaprakash Mutyala doBMCForceRestart(asyncResp); 137f92af389SJayaprakash Mutyala return; 138f92af389SJayaprakash Mutyala } 1392a5c4407SGunnar Mills BMCWEB_LOG_DEBUG << "Invalid property value for ResetType: " 1402a5c4407SGunnar Mills << resetType; 1412a5c4407SGunnar Mills messages::actionParameterNotSupported(asyncResp->res, resetType, 1422a5c4407SGunnar Mills "ResetType"); 1432a5c4407SGunnar Mills 1442a5c4407SGunnar Mills return; 1457e860f15SJohn Edward Broadbent }); 1462a5c4407SGunnar Mills } 147ed5befbdSJennifer Lee 1483e40fc74SGunnar Mills /** 1493e40fc74SGunnar Mills * ManagerResetToDefaultsAction class supports POST method for factory reset 1503e40fc74SGunnar Mills * action. 1513e40fc74SGunnar Mills */ 1527e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetToDefaultsAction(App& app) 1533e40fc74SGunnar Mills { 1543e40fc74SGunnar Mills 1553e40fc74SGunnar Mills /** 1563e40fc74SGunnar Mills * Function handles ResetToDefaults POST method request. 1573e40fc74SGunnar Mills * 1583e40fc74SGunnar Mills * Analyzes POST body message and factory resets BMC by calling 1593e40fc74SGunnar Mills * BMC code updater factory reset followed by a BMC reboot. 1603e40fc74SGunnar Mills * 1613e40fc74SGunnar Mills * BMC code updater factory reset wipes the whole BMC read-write 1623e40fc74SGunnar Mills * filesystem which includes things like the network settings. 1633e40fc74SGunnar Mills * 1643e40fc74SGunnar Mills * OpenBMC only supports ResetToDefaultsType "ResetAll". 1653e40fc74SGunnar Mills */ 1667e860f15SJohn Edward Broadbent 1677e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, 1687e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults/") 169ed398213SEd Tanous .privileges(redfish::privileges::postManager) 1707e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::post)( 1717e860f15SJohn Edward Broadbent [](const crow::Request& req, 1727e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 1733e40fc74SGunnar Mills BMCWEB_LOG_DEBUG << "Post ResetToDefaults."; 1743e40fc74SGunnar Mills 1753e40fc74SGunnar Mills std::string resetType; 1763e40fc74SGunnar Mills 17715ed6780SWilly Tu if (!json_util::readJsonAction( 17815ed6780SWilly Tu req, asyncResp->res, "ResetToDefaultsType", resetType)) 1793e40fc74SGunnar Mills { 1803e40fc74SGunnar Mills BMCWEB_LOG_DEBUG << "Missing property ResetToDefaultsType."; 1813e40fc74SGunnar Mills 1827e860f15SJohn Edward Broadbent messages::actionParameterMissing(asyncResp->res, 1837e860f15SJohn Edward Broadbent "ResetToDefaults", 1843e40fc74SGunnar Mills "ResetToDefaultsType"); 1853e40fc74SGunnar Mills return; 1863e40fc74SGunnar Mills } 1873e40fc74SGunnar Mills 1883e40fc74SGunnar Mills if (resetType != "ResetAll") 1893e40fc74SGunnar Mills { 1900fda0f12SGeorge Liu BMCWEB_LOG_DEBUG 1910fda0f12SGeorge Liu << "Invalid property value for ResetToDefaultsType: " 1923e40fc74SGunnar Mills << resetType; 1937e860f15SJohn Edward Broadbent messages::actionParameterNotSupported( 1947e860f15SJohn Edward Broadbent asyncResp->res, resetType, "ResetToDefaultsType"); 1953e40fc74SGunnar Mills return; 1963e40fc74SGunnar Mills } 1973e40fc74SGunnar Mills 1983e40fc74SGunnar Mills crow::connections::systemBus->async_method_call( 1993e40fc74SGunnar Mills [asyncResp](const boost::system::error_code ec) { 2003e40fc74SGunnar Mills if (ec) 2013e40fc74SGunnar Mills { 2027e860f15SJohn Edward Broadbent BMCWEB_LOG_DEBUG << "Failed to ResetToDefaults: " 2037e860f15SJohn Edward Broadbent << ec; 2043e40fc74SGunnar Mills messages::internalError(asyncResp->res); 2053e40fc74SGunnar Mills return; 2063e40fc74SGunnar Mills } 2073e40fc74SGunnar Mills // Factory Reset doesn't actually happen until a reboot 2083e40fc74SGunnar Mills // Can't erase what the BMC is running on 2093e40fc74SGunnar Mills doBMCGracefulRestart(asyncResp); 2103e40fc74SGunnar Mills }, 2113e40fc74SGunnar Mills "xyz.openbmc_project.Software.BMC.Updater", 2123e40fc74SGunnar Mills "/xyz/openbmc_project/software", 2133e40fc74SGunnar Mills "xyz.openbmc_project.Common.FactoryReset", "Reset"); 2147e860f15SJohn Edward Broadbent }); 2153e40fc74SGunnar Mills } 2163e40fc74SGunnar Mills 2171cb1a9e6SAppaRao Puli /** 2181cb1a9e6SAppaRao Puli * ManagerResetActionInfo derived class for delivering Manager 2191cb1a9e6SAppaRao Puli * ResetType AllowableValues using ResetInfo schema. 2201cb1a9e6SAppaRao Puli */ 2217e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetActionInfo(App& app) 2221cb1a9e6SAppaRao Puli { 2231cb1a9e6SAppaRao Puli /** 2241cb1a9e6SAppaRao Puli * Functions triggers appropriate requests on DBus 2251cb1a9e6SAppaRao Puli */ 2267e860f15SJohn Edward Broadbent 2277e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/ResetActionInfo/") 228ed398213SEd Tanous .privileges(redfish::privileges::getActionInfo) 2297e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)( 2307e860f15SJohn Edward Broadbent [](const crow::Request&, 2317e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 2328d1b46d7Szhanghch05 asyncResp->res.jsonValue = { 2331cb1a9e6SAppaRao Puli {"@odata.type", "#ActionInfo.v1_1_2.ActionInfo"}, 2341cb1a9e6SAppaRao Puli {"@odata.id", "/redfish/v1/Managers/bmc/ResetActionInfo"}, 2351cb1a9e6SAppaRao Puli {"Name", "Reset Action Info"}, 2361cb1a9e6SAppaRao Puli {"Id", "ResetActionInfo"}, 2371cb1a9e6SAppaRao Puli {"Parameters", 2381cb1a9e6SAppaRao Puli {{{"Name", "ResetType"}, 2391cb1a9e6SAppaRao Puli {"Required", true}, 2401cb1a9e6SAppaRao Puli {"DataType", "String"}, 2417e860f15SJohn Edward Broadbent {"AllowableValues", 2427e860f15SJohn Edward Broadbent {"GracefulRestart", "ForceRestart"}}}}}}; 2437e860f15SJohn Edward Broadbent }); 2441cb1a9e6SAppaRao Puli } 2451cb1a9e6SAppaRao Puli 2465b4aa86bSJames Feist static constexpr const char* objectManagerIface = 2475b4aa86bSJames Feist "org.freedesktop.DBus.ObjectManager"; 2485b4aa86bSJames Feist static constexpr const char* pidConfigurationIface = 2495b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid"; 2505b4aa86bSJames Feist static constexpr const char* pidZoneConfigurationIface = 2515b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid.Zone"; 252b7a08d04SJames Feist static constexpr const char* stepwiseConfigurationIface = 253b7a08d04SJames Feist "xyz.openbmc_project.Configuration.Stepwise"; 25473df0db0SJames Feist static constexpr const char* thermalModeIface = 25573df0db0SJames Feist "xyz.openbmc_project.Control.ThermalMode"; 2569c310685SBorawski.Lukasz 2578d1b46d7Szhanghch05 inline void 2588d1b46d7Szhanghch05 asyncPopulatePid(const std::string& connection, const std::string& path, 25973df0db0SJames Feist const std::string& currentProfile, 26073df0db0SJames Feist const std::vector<std::string>& supportedProfiles, 2618d1b46d7Szhanghch05 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 2625b4aa86bSJames Feist { 2635b4aa86bSJames Feist 2645b4aa86bSJames Feist crow::connections::systemBus->async_method_call( 26573df0db0SJames Feist [asyncResp, currentProfile, supportedProfiles]( 26673df0db0SJames Feist const boost::system::error_code ec, 2675b4aa86bSJames Feist const dbus::utility::ManagedObjectType& managedObj) { 2685b4aa86bSJames Feist if (ec) 2695b4aa86bSJames Feist { 2705b4aa86bSJames Feist BMCWEB_LOG_ERROR << ec; 2715b4aa86bSJames Feist asyncResp->res.jsonValue.clear(); 272f12894f8SJason M. Bills messages::internalError(asyncResp->res); 2735b4aa86bSJames Feist return; 2745b4aa86bSJames Feist } 2755b4aa86bSJames Feist nlohmann::json& configRoot = 2765b4aa86bSJames Feist asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"]; 2775b4aa86bSJames Feist nlohmann::json& fans = configRoot["FanControllers"]; 2785b4aa86bSJames Feist fans["@odata.type"] = "#OemManager.FanControllers"; 2790fda0f12SGeorge Liu fans["@odata.id"] = 2800fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanControllers"; 2815b4aa86bSJames Feist 2825b4aa86bSJames Feist nlohmann::json& pids = configRoot["PidControllers"]; 2835b4aa86bSJames Feist pids["@odata.type"] = "#OemManager.PidControllers"; 2845b4aa86bSJames Feist pids["@odata.id"] = 2855b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers"; 2865b4aa86bSJames Feist 287b7a08d04SJames Feist nlohmann::json& stepwise = configRoot["StepwiseControllers"]; 288b7a08d04SJames Feist stepwise["@odata.type"] = "#OemManager.StepwiseControllers"; 289b7a08d04SJames Feist stepwise["@odata.id"] = 290b7a08d04SJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers"; 291b7a08d04SJames Feist 2925b4aa86bSJames Feist nlohmann::json& zones = configRoot["FanZones"]; 2935b4aa86bSJames Feist zones["@odata.id"] = 2945b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones"; 2955b4aa86bSJames Feist zones["@odata.type"] = "#OemManager.FanZones"; 2965b4aa86bSJames Feist configRoot["@odata.id"] = 2975b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan"; 2985b4aa86bSJames Feist configRoot["@odata.type"] = "#OemManager.Fan"; 29973df0db0SJames Feist configRoot["Profile@Redfish.AllowableValues"] = supportedProfiles; 30073df0db0SJames Feist 30173df0db0SJames Feist if (!currentProfile.empty()) 30273df0db0SJames Feist { 30373df0db0SJames Feist configRoot["Profile"] = currentProfile; 30473df0db0SJames Feist } 30573df0db0SJames Feist BMCWEB_LOG_ERROR << "profile = " << currentProfile << " !"; 3065b4aa86bSJames Feist 3075b4aa86bSJames Feist for (const auto& pathPair : managedObj) 3085b4aa86bSJames Feist { 3095b4aa86bSJames Feist for (const auto& intfPair : pathPair.second) 3105b4aa86bSJames Feist { 3115b4aa86bSJames Feist if (intfPair.first != pidConfigurationIface && 312b7a08d04SJames Feist intfPair.first != pidZoneConfigurationIface && 313b7a08d04SJames Feist intfPair.first != stepwiseConfigurationIface) 3145b4aa86bSJames Feist { 3155b4aa86bSJames Feist continue; 3165b4aa86bSJames Feist } 31773df0db0SJames Feist 318711ac7a9SEd Tanous std::string name; 319711ac7a9SEd Tanous 320711ac7a9SEd Tanous for (const std::pair<std::string, 321711ac7a9SEd Tanous dbus::utility::DbusVariantType>& 322711ac7a9SEd Tanous propPair : intfPair.second) 323711ac7a9SEd Tanous { 324711ac7a9SEd Tanous if (propPair.first == "Name") 325711ac7a9SEd Tanous { 3265b4aa86bSJames Feist const std::string* namePtr = 327711ac7a9SEd Tanous std::get_if<std::string>(&propPair.second); 3285b4aa86bSJames Feist if (namePtr == nullptr) 3295b4aa86bSJames Feist { 3305b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Name Field illegal"; 331b7a08d04SJames Feist messages::internalError(asyncResp->res); 3325b4aa86bSJames Feist return; 3335b4aa86bSJames Feist } 334db697703SWilly Tu name = *namePtr; 3355b4aa86bSJames Feist dbus::utility::escapePathForDbus(name); 336711ac7a9SEd Tanous } 337711ac7a9SEd Tanous else if (propPair.first == "Profiles") 33873df0db0SJames Feist { 33973df0db0SJames Feist const std::vector<std::string>* profiles = 34073df0db0SJames Feist std::get_if<std::vector<std::string>>( 341711ac7a9SEd Tanous &propPair.second); 34273df0db0SJames Feist if (profiles == nullptr) 34373df0db0SJames Feist { 344711ac7a9SEd Tanous BMCWEB_LOG_ERROR 345711ac7a9SEd Tanous << "Pid Profiles Field illegal"; 34673df0db0SJames Feist messages::internalError(asyncResp->res); 34773df0db0SJames Feist return; 34873df0db0SJames Feist } 34973df0db0SJames Feist if (std::find(profiles->begin(), profiles->end(), 35073df0db0SJames Feist currentProfile) == profiles->end()) 35173df0db0SJames Feist { 35273df0db0SJames Feist BMCWEB_LOG_INFO 353711ac7a9SEd Tanous << name 354711ac7a9SEd Tanous << " not supported in current profile"; 35573df0db0SJames Feist continue; 35673df0db0SJames Feist } 35773df0db0SJames Feist } 358711ac7a9SEd Tanous } 359b7a08d04SJames Feist nlohmann::json* config = nullptr; 360c33a90ecSJames Feist const std::string* classPtr = nullptr; 361711ac7a9SEd Tanous 362711ac7a9SEd Tanous for (const std::pair<std::string, 363711ac7a9SEd Tanous dbus::utility::DbusVariantType>& 364711ac7a9SEd Tanous propPair : intfPair.second) 365c33a90ecSJames Feist { 366727dc83fSLei YU if (propPair.first == "Class") 367711ac7a9SEd Tanous { 368711ac7a9SEd Tanous classPtr = 369711ac7a9SEd Tanous std::get_if<std::string>(&propPair.second); 370711ac7a9SEd Tanous } 371c33a90ecSJames Feist } 372c33a90ecSJames Feist 3735b4aa86bSJames Feist if (intfPair.first == pidZoneConfigurationIface) 3745b4aa86bSJames Feist { 3755b4aa86bSJames Feist std::string chassis; 3765b4aa86bSJames Feist if (!dbus::utility::getNthStringFromPath( 3775b4aa86bSJames Feist pathPair.first.str, 5, chassis)) 3785b4aa86bSJames Feist { 3795b4aa86bSJames Feist chassis = "#IllegalValue"; 3805b4aa86bSJames Feist } 3815b4aa86bSJames Feist nlohmann::json& zone = zones[name]; 3825b4aa86bSJames Feist zone["Chassis"] = { 3835b4aa86bSJames Feist {"@odata.id", "/redfish/v1/Chassis/" + chassis}}; 3840fda0f12SGeorge Liu zone["@odata.id"] = 3850fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/" + 3865b4aa86bSJames Feist name; 3875b4aa86bSJames Feist zone["@odata.type"] = "#OemManager.FanZone"; 388b7a08d04SJames Feist config = &zone; 3895b4aa86bSJames Feist } 3905b4aa86bSJames Feist 391b7a08d04SJames Feist else if (intfPair.first == stepwiseConfigurationIface) 3925b4aa86bSJames Feist { 393c33a90ecSJames Feist if (classPtr == nullptr) 394c33a90ecSJames Feist { 395c33a90ecSJames Feist BMCWEB_LOG_ERROR << "Pid Class Field illegal"; 396c33a90ecSJames Feist messages::internalError(asyncResp->res); 397c33a90ecSJames Feist return; 398c33a90ecSJames Feist } 399c33a90ecSJames Feist 400b7a08d04SJames Feist nlohmann::json& controller = stepwise[name]; 401b7a08d04SJames Feist config = &controller; 4025b4aa86bSJames Feist 403b7a08d04SJames Feist controller["@odata.id"] = 4040fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers/" + 405271584abSEd Tanous name; 406b7a08d04SJames Feist controller["@odata.type"] = 407b7a08d04SJames Feist "#OemManager.StepwiseController"; 408b7a08d04SJames Feist 409c33a90ecSJames Feist controller["Direction"] = *classPtr; 4105b4aa86bSJames Feist } 4115b4aa86bSJames Feist 4125b4aa86bSJames Feist // pid and fans are off the same configuration 413b7a08d04SJames Feist else if (intfPair.first == pidConfigurationIface) 4145b4aa86bSJames Feist { 415c33a90ecSJames Feist 4165b4aa86bSJames Feist if (classPtr == nullptr) 4175b4aa86bSJames Feist { 4185b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Class Field illegal"; 419a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 4205b4aa86bSJames Feist return; 4215b4aa86bSJames Feist } 4225b4aa86bSJames Feist bool isFan = *classPtr == "fan"; 4235b4aa86bSJames Feist nlohmann::json& element = 4245b4aa86bSJames Feist isFan ? fans[name] : pids[name]; 425b7a08d04SJames Feist config = &element; 4265b4aa86bSJames Feist if (isFan) 4275b4aa86bSJames Feist { 4285b4aa86bSJames Feist element["@odata.id"] = 4290fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanControllers/" + 430271584abSEd Tanous name; 4315b4aa86bSJames Feist element["@odata.type"] = 4325b4aa86bSJames Feist "#OemManager.FanController"; 4335b4aa86bSJames Feist } 4345b4aa86bSJames Feist else 4355b4aa86bSJames Feist { 4365b4aa86bSJames Feist element["@odata.id"] = 4370fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers/" + 438271584abSEd Tanous name; 4395b4aa86bSJames Feist element["@odata.type"] = 4405b4aa86bSJames Feist "#OemManager.PidController"; 4415b4aa86bSJames Feist } 442b7a08d04SJames Feist } 443b7a08d04SJames Feist else 444b7a08d04SJames Feist { 445b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Unexpected configuration"; 446b7a08d04SJames Feist messages::internalError(asyncResp->res); 447b7a08d04SJames Feist return; 448b7a08d04SJames Feist } 449b7a08d04SJames Feist 450b7a08d04SJames Feist // used for making maps out of 2 vectors 451b7a08d04SJames Feist const std::vector<double>* keys = nullptr; 452b7a08d04SJames Feist const std::vector<double>* values = nullptr; 453b7a08d04SJames Feist 454b7a08d04SJames Feist for (const auto& propertyPair : intfPair.second) 455b7a08d04SJames Feist { 456b7a08d04SJames Feist if (propertyPair.first == "Type" || 457b7a08d04SJames Feist propertyPair.first == "Class" || 458b7a08d04SJames Feist propertyPair.first == "Name") 459b7a08d04SJames Feist { 460b7a08d04SJames Feist continue; 461b7a08d04SJames Feist } 462b7a08d04SJames Feist 463b7a08d04SJames Feist // zones 464b7a08d04SJames Feist if (intfPair.first == pidZoneConfigurationIface) 465b7a08d04SJames Feist { 466b7a08d04SJames Feist const double* ptr = 467abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 468b7a08d04SJames Feist if (ptr == nullptr) 469b7a08d04SJames Feist { 470b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 471b7a08d04SJames Feist << propertyPair.first; 472b7a08d04SJames Feist messages::internalError(asyncResp->res); 473b7a08d04SJames Feist return; 474b7a08d04SJames Feist } 475b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 476b7a08d04SJames Feist } 477b7a08d04SJames Feist 478b7a08d04SJames Feist if (intfPair.first == stepwiseConfigurationIface) 479b7a08d04SJames Feist { 480b7a08d04SJames Feist if (propertyPair.first == "Reading" || 481b7a08d04SJames Feist propertyPair.first == "Output") 482b7a08d04SJames Feist { 483b7a08d04SJames Feist const std::vector<double>* ptr = 484abf2add6SEd Tanous std::get_if<std::vector<double>>( 485b7a08d04SJames Feist &propertyPair.second); 486b7a08d04SJames Feist 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 495b7a08d04SJames Feist if (propertyPair.first == "Reading") 496b7a08d04SJames Feist { 497b7a08d04SJames Feist keys = ptr; 498b7a08d04SJames Feist } 499b7a08d04SJames Feist else 500b7a08d04SJames Feist { 501b7a08d04SJames Feist values = ptr; 502b7a08d04SJames Feist } 503e662eae8SEd Tanous if (keys != nullptr && values != nullptr) 504b7a08d04SJames Feist { 505b7a08d04SJames Feist if (keys->size() != values->size()) 506b7a08d04SJames Feist { 507b7a08d04SJames Feist BMCWEB_LOG_ERROR 5080fda0f12SGeorge Liu << "Reading and Output size don't match "; 509b7a08d04SJames Feist messages::internalError(asyncResp->res); 510b7a08d04SJames Feist return; 511b7a08d04SJames Feist } 512b7a08d04SJames Feist nlohmann::json& steps = (*config)["Steps"]; 513b7a08d04SJames Feist steps = nlohmann::json::array(); 514b7a08d04SJames Feist for (size_t ii = 0; ii < keys->size(); ii++) 515b7a08d04SJames Feist { 516b7a08d04SJames Feist steps.push_back( 517b7a08d04SJames Feist {{"Target", (*keys)[ii]}, 518b7a08d04SJames Feist {"Output", (*values)[ii]}}); 519b7a08d04SJames Feist } 520b7a08d04SJames Feist } 521b7a08d04SJames Feist } 522b7a08d04SJames Feist if (propertyPair.first == "NegativeHysteresis" || 523b7a08d04SJames Feist propertyPair.first == "PositiveHysteresis") 524b7a08d04SJames Feist { 525b7a08d04SJames Feist const double* ptr = 526abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 527b7a08d04SJames Feist if (ptr == nullptr) 528b7a08d04SJames Feist { 529b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 530b7a08d04SJames Feist << propertyPair.first; 531b7a08d04SJames Feist messages::internalError(asyncResp->res); 532b7a08d04SJames Feist return; 533b7a08d04SJames Feist } 534b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 535b7a08d04SJames Feist } 536b7a08d04SJames Feist } 537b7a08d04SJames Feist 538b7a08d04SJames Feist // pid and fans are off the same configuration 539b7a08d04SJames Feist if (intfPair.first == pidConfigurationIface || 540b7a08d04SJames Feist intfPair.first == stepwiseConfigurationIface) 541b7a08d04SJames Feist { 5425b4aa86bSJames Feist 5435b4aa86bSJames Feist if (propertyPair.first == "Zones") 5445b4aa86bSJames Feist { 5455b4aa86bSJames Feist const std::vector<std::string>* inputs = 546abf2add6SEd Tanous std::get_if<std::vector<std::string>>( 5471b6b96c5SEd Tanous &propertyPair.second); 5485b4aa86bSJames Feist 5495b4aa86bSJames Feist if (inputs == nullptr) 5505b4aa86bSJames Feist { 5515b4aa86bSJames Feist BMCWEB_LOG_ERROR 5525b4aa86bSJames Feist << "Zones Pid Field Illegal"; 553a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 5545b4aa86bSJames Feist return; 5555b4aa86bSJames Feist } 556b7a08d04SJames Feist auto& data = (*config)[propertyPair.first]; 5575b4aa86bSJames Feist data = nlohmann::json::array(); 5585b4aa86bSJames Feist for (std::string itemCopy : *inputs) 5595b4aa86bSJames Feist { 5605b4aa86bSJames Feist dbus::utility::escapePathForDbus(itemCopy); 5615b4aa86bSJames Feist data.push_back( 5625b4aa86bSJames Feist {{"@odata.id", 5630fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/" + 5645b4aa86bSJames Feist itemCopy}}); 5655b4aa86bSJames Feist } 5665b4aa86bSJames Feist } 5675b4aa86bSJames Feist // todo(james): may never happen, but this 5685b4aa86bSJames Feist // assumes configuration data referenced in the 5695b4aa86bSJames Feist // PID config is provided by the same daemon, we 5705b4aa86bSJames Feist // could add another loop to cover all cases, 5715b4aa86bSJames Feist // but I'm okay kicking this can down the road a 5725b4aa86bSJames Feist // bit 5735b4aa86bSJames Feist 5745b4aa86bSJames Feist else if (propertyPair.first == "Inputs" || 5755b4aa86bSJames Feist propertyPair.first == "Outputs") 5765b4aa86bSJames Feist { 577b7a08d04SJames Feist auto& data = (*config)[propertyPair.first]; 5785b4aa86bSJames Feist const std::vector<std::string>* inputs = 579abf2add6SEd Tanous std::get_if<std::vector<std::string>>( 5801b6b96c5SEd Tanous &propertyPair.second); 5815b4aa86bSJames Feist 5825b4aa86bSJames Feist if (inputs == nullptr) 5835b4aa86bSJames Feist { 5845b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 5855b4aa86bSJames Feist << propertyPair.first; 586f12894f8SJason M. Bills messages::internalError(asyncResp->res); 5875b4aa86bSJames Feist return; 5885b4aa86bSJames Feist } 5895b4aa86bSJames Feist data = *inputs; 590b943aaefSJames Feist } 591b943aaefSJames Feist else if (propertyPair.first == "SetPointOffset") 592b943aaefSJames Feist { 593b943aaefSJames Feist const std::string* ptr = 594b943aaefSJames Feist std::get_if<std::string>( 595b943aaefSJames Feist &propertyPair.second); 596b943aaefSJames Feist 597b943aaefSJames Feist if (ptr == nullptr) 598b943aaefSJames Feist { 599b943aaefSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 600b943aaefSJames Feist << propertyPair.first; 601b943aaefSJames Feist messages::internalError(asyncResp->res); 602b943aaefSJames Feist return; 603b943aaefSJames Feist } 604b943aaefSJames Feist // translate from dbus to redfish 605b943aaefSJames Feist if (*ptr == "WarningHigh") 606b943aaefSJames Feist { 607b943aaefSJames Feist (*config)["SetPointOffset"] = 608b943aaefSJames Feist "UpperThresholdNonCritical"; 609b943aaefSJames Feist } 610b943aaefSJames Feist else if (*ptr == "WarningLow") 611b943aaefSJames Feist { 612b943aaefSJames Feist (*config)["SetPointOffset"] = 613b943aaefSJames Feist "LowerThresholdNonCritical"; 614b943aaefSJames Feist } 615b943aaefSJames Feist else if (*ptr == "CriticalHigh") 616b943aaefSJames Feist { 617b943aaefSJames Feist (*config)["SetPointOffset"] = 618b943aaefSJames Feist "UpperThresholdCritical"; 619b943aaefSJames Feist } 620b943aaefSJames Feist else if (*ptr == "CriticalLow") 621b943aaefSJames Feist { 622b943aaefSJames Feist (*config)["SetPointOffset"] = 623b943aaefSJames Feist "LowerThresholdCritical"; 624b943aaefSJames Feist } 625b943aaefSJames Feist else 626b943aaefSJames Feist { 627b943aaefSJames Feist BMCWEB_LOG_ERROR << "Value Illegal " 628b943aaefSJames Feist << *ptr; 629b943aaefSJames Feist messages::internalError(asyncResp->res); 630b943aaefSJames Feist return; 631b943aaefSJames Feist } 632b943aaefSJames Feist } 633b943aaefSJames Feist // doubles 6345b4aa86bSJames Feist else if (propertyPair.first == 6355b4aa86bSJames Feist "FFGainCoefficient" || 6365b4aa86bSJames Feist propertyPair.first == "FFOffCoefficient" || 6375b4aa86bSJames Feist propertyPair.first == "ICoefficient" || 6385b4aa86bSJames Feist propertyPair.first == "ILimitMax" || 6395b4aa86bSJames Feist propertyPair.first == "ILimitMin" || 640aad1a257SJames Feist propertyPair.first == 641aad1a257SJames Feist "PositiveHysteresis" || 642aad1a257SJames Feist propertyPair.first == 643aad1a257SJames Feist "NegativeHysteresis" || 6445b4aa86bSJames Feist propertyPair.first == "OutLimitMax" || 6455b4aa86bSJames Feist propertyPair.first == "OutLimitMin" || 6465b4aa86bSJames Feist propertyPair.first == "PCoefficient" || 6477625cb81SJames Feist propertyPair.first == "SetPoint" || 6485b4aa86bSJames Feist propertyPair.first == "SlewNeg" || 6495b4aa86bSJames Feist propertyPair.first == "SlewPos") 6505b4aa86bSJames Feist { 6515b4aa86bSJames Feist const double* ptr = 652abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 6535b4aa86bSJames Feist if (ptr == nullptr) 6545b4aa86bSJames Feist { 6555b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 6565b4aa86bSJames Feist << propertyPair.first; 657f12894f8SJason M. Bills messages::internalError(asyncResp->res); 6585b4aa86bSJames Feist return; 6595b4aa86bSJames Feist } 660b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 6615b4aa86bSJames Feist } 6625b4aa86bSJames Feist } 6635b4aa86bSJames Feist } 6645b4aa86bSJames Feist } 6655b4aa86bSJames Feist } 6665b4aa86bSJames Feist }, 6675b4aa86bSJames Feist connection, path, objectManagerIface, "GetManagedObjects"); 6685b4aa86bSJames Feist } 669ca537928SJennifer Lee 67083ff9ab6SJames Feist enum class CreatePIDRet 67183ff9ab6SJames Feist { 67283ff9ab6SJames Feist fail, 67383ff9ab6SJames Feist del, 67483ff9ab6SJames Feist patch 67583ff9ab6SJames Feist }; 67683ff9ab6SJames Feist 6778d1b46d7Szhanghch05 inline bool 6788d1b46d7Szhanghch05 getZonesFromJsonReq(const std::shared_ptr<bmcweb::AsyncResp>& response, 6795f2caaefSJames Feist std::vector<nlohmann::json>& config, 6805f2caaefSJames Feist std::vector<std::string>& zones) 6815f2caaefSJames Feist { 682b6baeaa4SJames Feist if (config.empty()) 683b6baeaa4SJames Feist { 684b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Empty Zones"; 685b6baeaa4SJames Feist messages::propertyValueFormatError(response->res, 686b6baeaa4SJames Feist nlohmann::json::array(), "Zones"); 687b6baeaa4SJames Feist return false; 688b6baeaa4SJames Feist } 6895f2caaefSJames Feist for (auto& odata : config) 6905f2caaefSJames Feist { 6915f2caaefSJames Feist std::string path; 6925f2caaefSJames Feist if (!redfish::json_util::readJson(odata, response->res, "@odata.id", 6935f2caaefSJames Feist path)) 6945f2caaefSJames Feist { 6955f2caaefSJames Feist return false; 6965f2caaefSJames Feist } 6975f2caaefSJames Feist std::string input; 69861adbda3SJames Feist 69961adbda3SJames Feist // 8 below comes from 70061adbda3SJames Feist // /redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/Left 70161adbda3SJames Feist // 0 1 2 3 4 5 6 7 8 70261adbda3SJames Feist if (!dbus::utility::getNthStringFromPath(path, 8, input)) 7035f2caaefSJames Feist { 7045f2caaefSJames Feist BMCWEB_LOG_ERROR << "Got invalid path " << path; 7055f2caaefSJames Feist BMCWEB_LOG_ERROR << "Illegal Type Zones"; 7065f2caaefSJames Feist messages::propertyValueFormatError(response->res, odata.dump(), 7075f2caaefSJames Feist "Zones"); 7085f2caaefSJames Feist return false; 7095f2caaefSJames Feist } 7105f2caaefSJames Feist boost::replace_all(input, "_", " "); 7115f2caaefSJames Feist zones.emplace_back(std::move(input)); 7125f2caaefSJames Feist } 7135f2caaefSJames Feist return true; 7145f2caaefSJames Feist } 7155f2caaefSJames Feist 716711ac7a9SEd Tanous inline const dbus::utility::ManagedObjectType::value_type* 71773df0db0SJames Feist findChassis(const dbus::utility::ManagedObjectType& managedObj, 718b6baeaa4SJames Feist const std::string& value, std::string& chassis) 719b6baeaa4SJames Feist { 720b6baeaa4SJames Feist BMCWEB_LOG_DEBUG << "Find Chassis: " << value << "\n"; 721b6baeaa4SJames Feist 722b6baeaa4SJames Feist std::string escaped = boost::replace_all_copy(value, " ", "_"); 723b6baeaa4SJames Feist escaped = "/" + escaped; 724b6baeaa4SJames Feist auto it = std::find_if( 725b6baeaa4SJames Feist managedObj.begin(), managedObj.end(), [&escaped](const auto& obj) { 726b6baeaa4SJames Feist if (boost::algorithm::ends_with(obj.first.str, escaped)) 727b6baeaa4SJames Feist { 728b6baeaa4SJames Feist BMCWEB_LOG_DEBUG << "Matched " << obj.first.str << "\n"; 729b6baeaa4SJames Feist return true; 730b6baeaa4SJames Feist } 731b6baeaa4SJames Feist return false; 732b6baeaa4SJames Feist }); 733b6baeaa4SJames Feist 734b6baeaa4SJames Feist if (it == managedObj.end()) 735b6baeaa4SJames Feist { 73673df0db0SJames Feist return nullptr; 737b6baeaa4SJames Feist } 738b6baeaa4SJames Feist // 5 comes from <chassis-name> being the 5th element 739b6baeaa4SJames Feist // /xyz/openbmc_project/inventory/system/chassis/<chassis-name> 74073df0db0SJames Feist if (dbus::utility::getNthStringFromPath(it->first.str, 5, chassis)) 74173df0db0SJames Feist { 74273df0db0SJames Feist return &(*it); 74373df0db0SJames Feist } 74473df0db0SJames Feist 74573df0db0SJames Feist return nullptr; 746b6baeaa4SJames Feist } 747b6baeaa4SJames Feist 74823a21a1cSEd Tanous inline CreatePIDRet createPidInterface( 7498d1b46d7Szhanghch05 const std::shared_ptr<bmcweb::AsyncResp>& response, const std::string& type, 750b5a76932SEd Tanous const nlohmann::json::iterator& it, const std::string& path, 75183ff9ab6SJames Feist const dbus::utility::ManagedObjectType& managedObj, bool createNewObject, 75283ff9ab6SJames Feist boost::container::flat_map<std::string, dbus::utility::DbusVariantType>& 75383ff9ab6SJames Feist output, 75473df0db0SJames Feist std::string& chassis, const std::string& profile) 75583ff9ab6SJames Feist { 75683ff9ab6SJames Feist 7575f2caaefSJames Feist // common deleter 758b6baeaa4SJames Feist if (it.value() == nullptr) 7595f2caaefSJames Feist { 7605f2caaefSJames Feist std::string iface; 7615f2caaefSJames Feist if (type == "PidControllers" || type == "FanControllers") 7625f2caaefSJames Feist { 7635f2caaefSJames Feist iface = pidConfigurationIface; 7645f2caaefSJames Feist } 7655f2caaefSJames Feist else if (type == "FanZones") 7665f2caaefSJames Feist { 7675f2caaefSJames Feist iface = pidZoneConfigurationIface; 7685f2caaefSJames Feist } 7695f2caaefSJames Feist else if (type == "StepwiseControllers") 7705f2caaefSJames Feist { 7715f2caaefSJames Feist iface = stepwiseConfigurationIface; 7725f2caaefSJames Feist } 7735f2caaefSJames Feist else 7745f2caaefSJames Feist { 775a0744d38SGunnar Mills BMCWEB_LOG_ERROR << "Illegal Type " << type; 7765f2caaefSJames Feist messages::propertyUnknown(response->res, type); 7775f2caaefSJames Feist return CreatePIDRet::fail; 7785f2caaefSJames Feist } 7796ee7f774SJames Feist 7806ee7f774SJames Feist BMCWEB_LOG_DEBUG << "del " << path << " " << iface << "\n"; 7815f2caaefSJames Feist // delete interface 7825f2caaefSJames Feist crow::connections::systemBus->async_method_call( 7835f2caaefSJames Feist [response, path](const boost::system::error_code ec) { 7845f2caaefSJames Feist if (ec) 7855f2caaefSJames Feist { 7865f2caaefSJames Feist BMCWEB_LOG_ERROR << "Error patching " << path << ": " << ec; 7875f2caaefSJames Feist messages::internalError(response->res); 788b6baeaa4SJames Feist return; 7895f2caaefSJames Feist } 790b6baeaa4SJames Feist messages::success(response->res); 7915f2caaefSJames Feist }, 7925f2caaefSJames Feist "xyz.openbmc_project.EntityManager", path, iface, "Delete"); 7935f2caaefSJames Feist return CreatePIDRet::del; 7945f2caaefSJames Feist } 7955f2caaefSJames Feist 796711ac7a9SEd Tanous const dbus::utility::ManagedObjectType::value_type* managedItem = nullptr; 797b6baeaa4SJames Feist if (!createNewObject) 798b6baeaa4SJames Feist { 799b6baeaa4SJames Feist // if we aren't creating a new object, we should be able to find it on 800b6baeaa4SJames Feist // d-bus 80173df0db0SJames Feist managedItem = findChassis(managedObj, it.key(), chassis); 80273df0db0SJames Feist if (managedItem == nullptr) 803b6baeaa4SJames Feist { 804b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 805*ace85d60SEd Tanous messages::invalidObject(response->res, 806*ace85d60SEd Tanous crow::utility::urlFromPieces( 807*ace85d60SEd Tanous "redfish", "v1", "Chassis", chassis)); 808b6baeaa4SJames Feist return CreatePIDRet::fail; 809b6baeaa4SJames Feist } 810b6baeaa4SJames Feist } 811b6baeaa4SJames Feist 81226f6976fSEd Tanous if (!profile.empty() && 81373df0db0SJames Feist (type == "PidControllers" || type == "FanControllers" || 81473df0db0SJames Feist type == "StepwiseControllers")) 81573df0db0SJames Feist { 81673df0db0SJames Feist if (managedItem == nullptr) 81773df0db0SJames Feist { 81873df0db0SJames Feist output["Profiles"] = std::vector<std::string>{profile}; 81973df0db0SJames Feist } 82073df0db0SJames Feist else 82173df0db0SJames Feist { 82273df0db0SJames Feist std::string interface; 82373df0db0SJames Feist if (type == "StepwiseControllers") 82473df0db0SJames Feist { 82573df0db0SJames Feist interface = stepwiseConfigurationIface; 82673df0db0SJames Feist } 82773df0db0SJames Feist else 82873df0db0SJames Feist { 82973df0db0SJames Feist interface = pidConfigurationIface; 83073df0db0SJames Feist } 831711ac7a9SEd Tanous bool ifaceFound = false; 832711ac7a9SEd Tanous for (const auto& iface : managedItem->second) 833711ac7a9SEd Tanous { 834711ac7a9SEd Tanous if (iface.first == interface) 835711ac7a9SEd Tanous { 836711ac7a9SEd Tanous ifaceFound = true; 837711ac7a9SEd Tanous for (const auto& prop : iface.second) 838711ac7a9SEd Tanous { 839711ac7a9SEd Tanous if (prop.first == "Profiles") 840711ac7a9SEd Tanous { 841711ac7a9SEd Tanous const std::vector<std::string>* curProfiles = 842711ac7a9SEd Tanous std::get_if<std::vector<std::string>>( 843711ac7a9SEd Tanous &(prop.second)); 844711ac7a9SEd Tanous if (curProfiles == nullptr) 845711ac7a9SEd Tanous { 846711ac7a9SEd Tanous BMCWEB_LOG_ERROR 847711ac7a9SEd Tanous << "Illegal profiles in managed object"; 848711ac7a9SEd Tanous messages::internalError(response->res); 849711ac7a9SEd Tanous return CreatePIDRet::fail; 850711ac7a9SEd Tanous } 851711ac7a9SEd Tanous if (std::find(curProfiles->begin(), 852711ac7a9SEd Tanous curProfiles->end(), 853711ac7a9SEd Tanous profile) == curProfiles->end()) 854711ac7a9SEd Tanous { 855711ac7a9SEd Tanous std::vector<std::string> newProfiles = 856711ac7a9SEd Tanous *curProfiles; 857711ac7a9SEd Tanous newProfiles.push_back(profile); 858711ac7a9SEd Tanous output["Profiles"] = newProfiles; 859711ac7a9SEd Tanous } 860711ac7a9SEd Tanous } 861711ac7a9SEd Tanous } 862711ac7a9SEd Tanous } 863711ac7a9SEd Tanous } 864711ac7a9SEd Tanous 865711ac7a9SEd Tanous if (!ifaceFound) 86673df0db0SJames Feist { 86773df0db0SJames Feist BMCWEB_LOG_ERROR 86873df0db0SJames Feist << "Failed to find interface in managed object"; 86973df0db0SJames Feist messages::internalError(response->res); 87073df0db0SJames Feist return CreatePIDRet::fail; 87173df0db0SJames Feist } 87273df0db0SJames Feist } 87373df0db0SJames Feist } 87473df0db0SJames Feist 87583ff9ab6SJames Feist if (type == "PidControllers" || type == "FanControllers") 87683ff9ab6SJames Feist { 87783ff9ab6SJames Feist if (createNewObject) 87883ff9ab6SJames Feist { 87983ff9ab6SJames Feist output["Class"] = type == "PidControllers" ? std::string("temp") 88083ff9ab6SJames Feist : std::string("fan"); 88183ff9ab6SJames Feist output["Type"] = std::string("Pid"); 88283ff9ab6SJames Feist } 8835f2caaefSJames Feist 8845f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> zones; 8855f2caaefSJames Feist std::optional<std::vector<std::string>> inputs; 8865f2caaefSJames Feist std::optional<std::vector<std::string>> outputs; 8875f2caaefSJames Feist std::map<std::string, std::optional<double>> doubles; 888b943aaefSJames Feist std::optional<std::string> setpointOffset; 8895f2caaefSJames Feist if (!redfish::json_util::readJson( 890b6baeaa4SJames Feist it.value(), response->res, "Inputs", inputs, "Outputs", outputs, 8915f2caaefSJames Feist "Zones", zones, "FFGainCoefficient", 8925f2caaefSJames Feist doubles["FFGainCoefficient"], "FFOffCoefficient", 8935f2caaefSJames Feist doubles["FFOffCoefficient"], "ICoefficient", 8945f2caaefSJames Feist doubles["ICoefficient"], "ILimitMax", doubles["ILimitMax"], 8955f2caaefSJames Feist "ILimitMin", doubles["ILimitMin"], "OutLimitMax", 8965f2caaefSJames Feist doubles["OutLimitMax"], "OutLimitMin", doubles["OutLimitMin"], 8975f2caaefSJames Feist "PCoefficient", doubles["PCoefficient"], "SetPoint", 898b943aaefSJames Feist doubles["SetPoint"], "SetPointOffset", setpointOffset, 899b943aaefSJames Feist "SlewNeg", doubles["SlewNeg"], "SlewPos", doubles["SlewPos"], 900b943aaefSJames Feist "PositiveHysteresis", doubles["PositiveHysteresis"], 901b943aaefSJames Feist "NegativeHysteresis", doubles["NegativeHysteresis"])) 90283ff9ab6SJames Feist { 90371f52d96SEd Tanous BMCWEB_LOG_ERROR 90471f52d96SEd Tanous << "Illegal Property " 90571f52d96SEd Tanous << it.value().dump(2, ' ', true, 90671f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 9075f2caaefSJames Feist return CreatePIDRet::fail; 90883ff9ab6SJames Feist } 9095f2caaefSJames Feist if (zones) 9105f2caaefSJames Feist { 9115f2caaefSJames Feist std::vector<std::string> zonesStr; 9125f2caaefSJames Feist if (!getZonesFromJsonReq(response, *zones, zonesStr)) 9135f2caaefSJames Feist { 914a0744d38SGunnar Mills BMCWEB_LOG_ERROR << "Illegal Zones"; 9155f2caaefSJames Feist return CreatePIDRet::fail; 9165f2caaefSJames Feist } 917b6baeaa4SJames Feist if (chassis.empty() && 918e662eae8SEd Tanous findChassis(managedObj, zonesStr[0], chassis) == nullptr) 919b6baeaa4SJames Feist { 920b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 921*ace85d60SEd Tanous messages::invalidObject( 922*ace85d60SEd Tanous response->res, crow::utility::urlFromPieces( 923*ace85d60SEd Tanous "redfish", "v1", "Chassis", chassis)); 924b6baeaa4SJames Feist return CreatePIDRet::fail; 925b6baeaa4SJames Feist } 926b6baeaa4SJames Feist 9275f2caaefSJames Feist output["Zones"] = std::move(zonesStr); 9285f2caaefSJames Feist } 9295f2caaefSJames Feist if (inputs || outputs) 9305f2caaefSJames Feist { 9315f2caaefSJames Feist std::array<std::optional<std::vector<std::string>>*, 2> containers = 9325f2caaefSJames Feist {&inputs, &outputs}; 9335f2caaefSJames Feist size_t index = 0; 9345f2caaefSJames Feist for (const auto& containerPtr : containers) 9355f2caaefSJames Feist { 9365f2caaefSJames Feist std::optional<std::vector<std::string>>& container = 9375f2caaefSJames Feist *containerPtr; 9385f2caaefSJames Feist if (!container) 9395f2caaefSJames Feist { 9405f2caaefSJames Feist index++; 9415f2caaefSJames Feist continue; 94283ff9ab6SJames Feist } 94383ff9ab6SJames Feist 9445f2caaefSJames Feist for (std::string& value : *container) 94583ff9ab6SJames Feist { 9465f2caaefSJames Feist boost::replace_all(value, "_", " "); 94783ff9ab6SJames Feist } 9485f2caaefSJames Feist std::string key; 9495f2caaefSJames Feist if (index == 0) 9505f2caaefSJames Feist { 9515f2caaefSJames Feist key = "Inputs"; 9525f2caaefSJames Feist } 9535f2caaefSJames Feist else 9545f2caaefSJames Feist { 9555f2caaefSJames Feist key = "Outputs"; 9565f2caaefSJames Feist } 9575f2caaefSJames Feist output[key] = *container; 9585f2caaefSJames Feist index++; 9595f2caaefSJames Feist } 96083ff9ab6SJames Feist } 96183ff9ab6SJames Feist 962b943aaefSJames Feist if (setpointOffset) 963b943aaefSJames Feist { 964b943aaefSJames Feist // translate between redfish and dbus names 965b943aaefSJames Feist if (*setpointOffset == "UpperThresholdNonCritical") 966b943aaefSJames Feist { 967b943aaefSJames Feist output["SetPointOffset"] = std::string("WarningLow"); 968b943aaefSJames Feist } 969b943aaefSJames Feist else if (*setpointOffset == "LowerThresholdNonCritical") 970b943aaefSJames Feist { 971b943aaefSJames Feist output["SetPointOffset"] = std::string("WarningHigh"); 972b943aaefSJames Feist } 973b943aaefSJames Feist else if (*setpointOffset == "LowerThresholdCritical") 974b943aaefSJames Feist { 975b943aaefSJames Feist output["SetPointOffset"] = std::string("CriticalLow"); 976b943aaefSJames Feist } 977b943aaefSJames Feist else if (*setpointOffset == "UpperThresholdCritical") 978b943aaefSJames Feist { 979b943aaefSJames Feist output["SetPointOffset"] = std::string("CriticalHigh"); 980b943aaefSJames Feist } 981b943aaefSJames Feist else 982b943aaefSJames Feist { 983b943aaefSJames Feist BMCWEB_LOG_ERROR << "Invalid setpointoffset " 984b943aaefSJames Feist << *setpointOffset; 985*ace85d60SEd Tanous messages::propertyValueNotInList(response->res, it.key(), 986*ace85d60SEd Tanous "SetPointOffset"); 987b943aaefSJames Feist return CreatePIDRet::fail; 988b943aaefSJames Feist } 989b943aaefSJames Feist } 990b943aaefSJames Feist 99183ff9ab6SJames Feist // doubles 9925f2caaefSJames Feist for (const auto& pairs : doubles) 99383ff9ab6SJames Feist { 9945f2caaefSJames Feist if (!pairs.second) 99583ff9ab6SJames Feist { 9965f2caaefSJames Feist continue; 99783ff9ab6SJames Feist } 9985f2caaefSJames Feist BMCWEB_LOG_DEBUG << pairs.first << " = " << *pairs.second; 9995f2caaefSJames Feist output[pairs.first] = *(pairs.second); 10005f2caaefSJames Feist } 100183ff9ab6SJames Feist } 100283ff9ab6SJames Feist 100383ff9ab6SJames Feist else if (type == "FanZones") 100483ff9ab6SJames Feist { 100583ff9ab6SJames Feist output["Type"] = std::string("Pid.Zone"); 100683ff9ab6SJames Feist 10075f2caaefSJames Feist std::optional<nlohmann::json> chassisContainer; 10085f2caaefSJames Feist std::optional<double> failSafePercent; 1009d3ec07f8SJames Feist std::optional<double> minThermalOutput; 1010b6baeaa4SJames Feist if (!redfish::json_util::readJson(it.value(), response->res, "Chassis", 10115f2caaefSJames Feist chassisContainer, "FailSafePercent", 1012d3ec07f8SJames Feist failSafePercent, "MinThermalOutput", 1013d3ec07f8SJames Feist minThermalOutput)) 101483ff9ab6SJames Feist { 101571f52d96SEd Tanous BMCWEB_LOG_ERROR 101671f52d96SEd Tanous << "Illegal Property " 101771f52d96SEd Tanous << it.value().dump(2, ' ', true, 101871f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 101983ff9ab6SJames Feist return CreatePIDRet::fail; 102083ff9ab6SJames Feist } 10215f2caaefSJames Feist 10225f2caaefSJames Feist if (chassisContainer) 102383ff9ab6SJames Feist { 10245f2caaefSJames Feist 10255f2caaefSJames Feist std::string chassisId; 10265f2caaefSJames Feist if (!redfish::json_util::readJson(*chassisContainer, response->res, 10275f2caaefSJames Feist "@odata.id", chassisId)) 10285f2caaefSJames Feist { 102971f52d96SEd Tanous BMCWEB_LOG_ERROR 103071f52d96SEd Tanous << "Illegal Property " 103171f52d96SEd Tanous << chassisContainer->dump( 103271f52d96SEd Tanous 2, ' ', true, 103371f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 103483ff9ab6SJames Feist return CreatePIDRet::fail; 103583ff9ab6SJames Feist } 103683ff9ab6SJames Feist 1037717794d5SAppaRao Puli // /redfish/v1/chassis/chassis_name/ 10385f2caaefSJames Feist if (!dbus::utility::getNthStringFromPath(chassisId, 3, chassis)) 103983ff9ab6SJames Feist { 10405f2caaefSJames Feist BMCWEB_LOG_ERROR << "Got invalid path " << chassisId; 1041*ace85d60SEd Tanous messages::invalidObject( 1042*ace85d60SEd Tanous response->res, crow::utility::urlFromPieces( 1043*ace85d60SEd Tanous "redfish", "v1", "Chassis", chassisId)); 104483ff9ab6SJames Feist return CreatePIDRet::fail; 104583ff9ab6SJames Feist } 104683ff9ab6SJames Feist } 1047d3ec07f8SJames Feist if (minThermalOutput) 104883ff9ab6SJames Feist { 1049d3ec07f8SJames Feist output["MinThermalOutput"] = *minThermalOutput; 10505f2caaefSJames Feist } 10515f2caaefSJames Feist if (failSafePercent) 105283ff9ab6SJames Feist { 10535f2caaefSJames Feist output["FailSafePercent"] = *failSafePercent; 10545f2caaefSJames Feist } 10555f2caaefSJames Feist } 10565f2caaefSJames Feist else if (type == "StepwiseControllers") 10575f2caaefSJames Feist { 10585f2caaefSJames Feist output["Type"] = std::string("Stepwise"); 10595f2caaefSJames Feist 10605f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> zones; 10615f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> steps; 10625f2caaefSJames Feist std::optional<std::vector<std::string>> inputs; 10635f2caaefSJames Feist std::optional<double> positiveHysteresis; 10645f2caaefSJames Feist std::optional<double> negativeHysteresis; 1065c33a90ecSJames Feist std::optional<std::string> direction; // upper clipping curve vs lower 10665f2caaefSJames Feist if (!redfish::json_util::readJson( 1067b6baeaa4SJames Feist it.value(), response->res, "Zones", zones, "Steps", steps, 1068b6baeaa4SJames Feist "Inputs", inputs, "PositiveHysteresis", positiveHysteresis, 1069c33a90ecSJames Feist "NegativeHysteresis", negativeHysteresis, "Direction", 1070c33a90ecSJames Feist direction)) 10715f2caaefSJames Feist { 107271f52d96SEd Tanous BMCWEB_LOG_ERROR 107371f52d96SEd Tanous << "Illegal Property " 107471f52d96SEd Tanous << it.value().dump(2, ' ', true, 107571f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 107683ff9ab6SJames Feist return CreatePIDRet::fail; 107783ff9ab6SJames Feist } 10785f2caaefSJames Feist 10795f2caaefSJames Feist if (zones) 108083ff9ab6SJames Feist { 1081b6baeaa4SJames Feist std::vector<std::string> zonesStrs; 1082b6baeaa4SJames Feist if (!getZonesFromJsonReq(response, *zones, zonesStrs)) 10835f2caaefSJames Feist { 1084a0744d38SGunnar Mills BMCWEB_LOG_ERROR << "Illegal Zones"; 108583ff9ab6SJames Feist return CreatePIDRet::fail; 108683ff9ab6SJames Feist } 1087b6baeaa4SJames Feist if (chassis.empty() && 1088e662eae8SEd Tanous findChassis(managedObj, zonesStrs[0], chassis) == nullptr) 1089b6baeaa4SJames Feist { 1090b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 1091*ace85d60SEd Tanous messages::invalidObject( 1092*ace85d60SEd Tanous response->res, crow::utility::urlFromPieces( 1093*ace85d60SEd Tanous "redfish", "v1", "Chassis", chassis)); 1094b6baeaa4SJames Feist return CreatePIDRet::fail; 1095b6baeaa4SJames Feist } 1096b6baeaa4SJames Feist output["Zones"] = std::move(zonesStrs); 10975f2caaefSJames Feist } 10985f2caaefSJames Feist if (steps) 10995f2caaefSJames Feist { 11005f2caaefSJames Feist std::vector<double> readings; 11015f2caaefSJames Feist std::vector<double> outputs; 11025f2caaefSJames Feist for (auto& step : *steps) 11035f2caaefSJames Feist { 1104543f4400SEd Tanous double target = 0.0; 1105543f4400SEd Tanous double out = 0.0; 11065f2caaefSJames Feist 11075f2caaefSJames Feist if (!redfish::json_util::readJson(step, response->res, "Target", 110823a21a1cSEd Tanous target, "Output", out)) 11095f2caaefSJames Feist { 111071f52d96SEd Tanous BMCWEB_LOG_ERROR 111171f52d96SEd Tanous << "Illegal Property " 111271f52d96SEd Tanous << it.value().dump( 111371f52d96SEd Tanous 2, ' ', true, 111471f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 11155f2caaefSJames Feist return CreatePIDRet::fail; 11165f2caaefSJames Feist } 11175f2caaefSJames Feist readings.emplace_back(target); 111823a21a1cSEd Tanous outputs.emplace_back(out); 11195f2caaefSJames Feist } 11205f2caaefSJames Feist output["Reading"] = std::move(readings); 11215f2caaefSJames Feist output["Output"] = std::move(outputs); 11225f2caaefSJames Feist } 11235f2caaefSJames Feist if (inputs) 11245f2caaefSJames Feist { 11255f2caaefSJames Feist for (std::string& value : *inputs) 11265f2caaefSJames Feist { 11275f2caaefSJames Feist boost::replace_all(value, "_", " "); 11285f2caaefSJames Feist } 11295f2caaefSJames Feist output["Inputs"] = std::move(*inputs); 11305f2caaefSJames Feist } 11315f2caaefSJames Feist if (negativeHysteresis) 11325f2caaefSJames Feist { 11335f2caaefSJames Feist output["NegativeHysteresis"] = *negativeHysteresis; 11345f2caaefSJames Feist } 11355f2caaefSJames Feist if (positiveHysteresis) 11365f2caaefSJames Feist { 11375f2caaefSJames Feist output["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 } 1150c33a90ecSJames Feist output["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 { 116383ff9ab6SJames Feist 11648d1b46d7Szhanghch05 GetPIDValues(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn) : 116523a21a1cSEd Tanous asyncResp(asyncRespIn) 116673df0db0SJames Feist 11671214b7e7SGunnar Mills {} 11689c310685SBorawski.Lukasz 116973df0db0SJames Feist void run() 11705b4aa86bSJames Feist { 117173df0db0SJames Feist std::shared_ptr<GetPIDValues> self = shared_from_this(); 117273df0db0SJames Feist 117373df0db0SJames Feist // get all configurations 11745b4aa86bSJames Feist crow::connections::systemBus->async_method_call( 117573df0db0SJames Feist [self](const boost::system::error_code ec, 117623a21a1cSEd Tanous const crow::openbmc_mapper::GetSubTreeType& subtreeLocal) { 11775b4aa86bSJames Feist if (ec) 11785b4aa86bSJames Feist { 11795b4aa86bSJames Feist BMCWEB_LOG_ERROR << ec; 118073df0db0SJames Feist messages::internalError(self->asyncResp->res); 118173df0db0SJames Feist return; 118273df0db0SJames Feist } 118323a21a1cSEd Tanous self->subtree = subtreeLocal; 118473df0db0SJames Feist }, 118573df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", 118673df0db0SJames Feist "/xyz/openbmc_project/object_mapper", 118773df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 118873df0db0SJames Feist std::array<const char*, 4>{ 118973df0db0SJames Feist pidConfigurationIface, pidZoneConfigurationIface, 119073df0db0SJames Feist objectManagerIface, stepwiseConfigurationIface}); 119173df0db0SJames Feist 119273df0db0SJames Feist // at the same time get the selected profile 119373df0db0SJames Feist crow::connections::systemBus->async_method_call( 119473df0db0SJames Feist [self](const boost::system::error_code ec, 119523a21a1cSEd Tanous const crow::openbmc_mapper::GetSubTreeType& subtreeLocal) { 119623a21a1cSEd Tanous if (ec || subtreeLocal.empty()) 119773df0db0SJames Feist { 119873df0db0SJames Feist return; 119973df0db0SJames Feist } 120023a21a1cSEd Tanous if (subtreeLocal[0].second.size() != 1) 120173df0db0SJames Feist { 120273df0db0SJames Feist // invalid mapper response, should never happen 120373df0db0SJames Feist BMCWEB_LOG_ERROR << "GetPIDValues: Mapper Error"; 120473df0db0SJames Feist messages::internalError(self->asyncResp->res); 12055b4aa86bSJames Feist return; 12065b4aa86bSJames Feist } 12075b4aa86bSJames Feist 120823a21a1cSEd Tanous const std::string& path = subtreeLocal[0].first; 120923a21a1cSEd Tanous const std::string& owner = subtreeLocal[0].second[0].first; 121073df0db0SJames Feist crow::connections::systemBus->async_method_call( 1211168e20c1SEd Tanous [path, owner, 1212168e20c1SEd Tanous self](const boost::system::error_code ec2, 121373df0db0SJames Feist const boost::container::flat_map< 1214168e20c1SEd Tanous std::string, dbus::utility::DbusVariantType>& 1215168e20c1SEd Tanous resp) { 121623a21a1cSEd Tanous if (ec2) 121773df0db0SJames Feist { 12180fda0f12SGeorge Liu BMCWEB_LOG_ERROR 12190fda0f12SGeorge Liu << "GetPIDValues: Can't get thermalModeIface " 122073df0db0SJames Feist << path; 122173df0db0SJames Feist messages::internalError(self->asyncResp->res); 122273df0db0SJames Feist return; 122373df0db0SJames Feist } 1224271584abSEd Tanous const std::string* current = nullptr; 1225271584abSEd Tanous const std::vector<std::string>* supported = nullptr; 12269eb808c1SEd Tanous for (const auto& [key, value] : resp) 122773df0db0SJames Feist { 122873df0db0SJames Feist if (key == "Current") 122973df0db0SJames Feist { 123073df0db0SJames Feist current = std::get_if<std::string>(&value); 123173df0db0SJames Feist if (current == nullptr) 123273df0db0SJames Feist { 123373df0db0SJames Feist BMCWEB_LOG_ERROR 12340fda0f12SGeorge Liu << "GetPIDValues: thermal mode iface invalid " 123573df0db0SJames Feist << path; 123673df0db0SJames Feist messages::internalError( 123773df0db0SJames Feist self->asyncResp->res); 123873df0db0SJames Feist return; 123973df0db0SJames Feist } 124073df0db0SJames Feist } 124173df0db0SJames Feist if (key == "Supported") 124273df0db0SJames Feist { 124373df0db0SJames Feist supported = 124473df0db0SJames Feist std::get_if<std::vector<std::string>>( 124573df0db0SJames Feist &value); 124673df0db0SJames Feist if (supported == nullptr) 124773df0db0SJames Feist { 124873df0db0SJames Feist BMCWEB_LOG_ERROR 12490fda0f12SGeorge Liu << "GetPIDValues: thermal mode iface invalid" 125073df0db0SJames Feist << path; 125173df0db0SJames Feist messages::internalError( 125273df0db0SJames Feist self->asyncResp->res); 125373df0db0SJames Feist return; 125473df0db0SJames Feist } 125573df0db0SJames Feist } 125673df0db0SJames Feist } 125773df0db0SJames Feist if (current == nullptr || supported == nullptr) 125873df0db0SJames Feist { 12590fda0f12SGeorge Liu BMCWEB_LOG_ERROR 12600fda0f12SGeorge Liu << "GetPIDValues: thermal mode iface invalid " 126173df0db0SJames Feist << path; 126273df0db0SJames Feist messages::internalError(self->asyncResp->res); 126373df0db0SJames Feist return; 126473df0db0SJames Feist } 126573df0db0SJames Feist self->currentProfile = *current; 126673df0db0SJames Feist self->supportedProfiles = *supported; 126773df0db0SJames Feist }, 126873df0db0SJames Feist owner, path, "org.freedesktop.DBus.Properties", "GetAll", 126973df0db0SJames Feist thermalModeIface); 127073df0db0SJames Feist }, 127173df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", 127273df0db0SJames Feist "/xyz/openbmc_project/object_mapper", 127373df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 127473df0db0SJames Feist std::array<const char*, 1>{thermalModeIface}); 127573df0db0SJames Feist } 127673df0db0SJames Feist 127773df0db0SJames Feist ~GetPIDValues() 127873df0db0SJames Feist { 127973df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 128073df0db0SJames Feist { 128173df0db0SJames Feist return; 128273df0db0SJames Feist } 12835b4aa86bSJames Feist // create map of <connection, path to objMgr>> 128473df0db0SJames Feist boost::container::flat_map<std::string, std::string> objectMgrPaths; 12856bce33bcSJames Feist boost::container::flat_set<std::string> calledConnections; 12865b4aa86bSJames Feist for (const auto& pathGroup : subtree) 12875b4aa86bSJames Feist { 12885b4aa86bSJames Feist for (const auto& connectionGroup : pathGroup.second) 12895b4aa86bSJames Feist { 12906bce33bcSJames Feist auto findConnection = 12916bce33bcSJames Feist calledConnections.find(connectionGroup.first); 12926bce33bcSJames Feist if (findConnection != calledConnections.end()) 12936bce33bcSJames Feist { 12946bce33bcSJames Feist break; 12956bce33bcSJames Feist } 129673df0db0SJames Feist for (const std::string& interface : connectionGroup.second) 12975b4aa86bSJames Feist { 12985b4aa86bSJames Feist if (interface == objectManagerIface) 12995b4aa86bSJames Feist { 130073df0db0SJames Feist objectMgrPaths[connectionGroup.first] = pathGroup.first; 13015b4aa86bSJames Feist } 13025b4aa86bSJames Feist // this list is alphabetical, so we 13035b4aa86bSJames Feist // should have found the objMgr by now 13045b4aa86bSJames Feist if (interface == pidConfigurationIface || 1305b7a08d04SJames Feist interface == pidZoneConfigurationIface || 1306b7a08d04SJames Feist interface == stepwiseConfigurationIface) 13075b4aa86bSJames Feist { 13085b4aa86bSJames Feist auto findObjMgr = 13095b4aa86bSJames Feist objectMgrPaths.find(connectionGroup.first); 13105b4aa86bSJames Feist if (findObjMgr == objectMgrPaths.end()) 13115b4aa86bSJames Feist { 13125b4aa86bSJames Feist BMCWEB_LOG_DEBUG << connectionGroup.first 13135b4aa86bSJames Feist << "Has no Object Manager"; 13145b4aa86bSJames Feist continue; 13155b4aa86bSJames Feist } 13166bce33bcSJames Feist 13176bce33bcSJames Feist calledConnections.insert(connectionGroup.first); 13186bce33bcSJames Feist 131973df0db0SJames Feist asyncPopulatePid(findObjMgr->first, findObjMgr->second, 132073df0db0SJames Feist currentProfile, supportedProfiles, 132173df0db0SJames Feist asyncResp); 13225b4aa86bSJames Feist break; 13235b4aa86bSJames Feist } 13245b4aa86bSJames Feist } 13255b4aa86bSJames Feist } 13265b4aa86bSJames Feist } 132773df0db0SJames Feist } 132873df0db0SJames Feist 1329ecd6a3a2SEd Tanous GetPIDValues(const GetPIDValues&) = delete; 1330ecd6a3a2SEd Tanous GetPIDValues(GetPIDValues&&) = delete; 1331ecd6a3a2SEd Tanous GetPIDValues& operator=(const GetPIDValues&) = delete; 1332ecd6a3a2SEd Tanous GetPIDValues& operator=(GetPIDValues&&) = delete; 1333ecd6a3a2SEd Tanous 133473df0db0SJames Feist std::vector<std::string> supportedProfiles; 133573df0db0SJames Feist std::string currentProfile; 133673df0db0SJames Feist crow::openbmc_mapper::GetSubTreeType subtree; 13378d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> asyncResp; 133873df0db0SJames Feist }; 133973df0db0SJames Feist 134073df0db0SJames Feist struct SetPIDValues : std::enable_shared_from_this<SetPIDValues> 134173df0db0SJames Feist { 134273df0db0SJames Feist 13438d1b46d7Szhanghch05 SetPIDValues(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn, 134473df0db0SJames Feist nlohmann::json& data) : 1345271584abSEd Tanous asyncResp(asyncRespIn) 134673df0db0SJames Feist { 134773df0db0SJames Feist 134873df0db0SJames Feist std::optional<nlohmann::json> pidControllers; 134973df0db0SJames Feist std::optional<nlohmann::json> fanControllers; 135073df0db0SJames Feist std::optional<nlohmann::json> fanZones; 135173df0db0SJames Feist std::optional<nlohmann::json> stepwiseControllers; 135273df0db0SJames Feist 135373df0db0SJames Feist if (!redfish::json_util::readJson( 135473df0db0SJames Feist data, asyncResp->res, "PidControllers", pidControllers, 135573df0db0SJames Feist "FanControllers", fanControllers, "FanZones", fanZones, 135673df0db0SJames Feist "StepwiseControllers", stepwiseControllers, "Profile", profile)) 135773df0db0SJames Feist { 135871f52d96SEd Tanous BMCWEB_LOG_ERROR 135971f52d96SEd Tanous << "Illegal Property " 136071f52d96SEd Tanous << data.dump(2, ' ', true, 136171f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 136273df0db0SJames Feist return; 136373df0db0SJames Feist } 136473df0db0SJames Feist configuration.emplace_back("PidControllers", std::move(pidControllers)); 136573df0db0SJames Feist configuration.emplace_back("FanControllers", std::move(fanControllers)); 136673df0db0SJames Feist configuration.emplace_back("FanZones", std::move(fanZones)); 136773df0db0SJames Feist configuration.emplace_back("StepwiseControllers", 136873df0db0SJames Feist std::move(stepwiseControllers)); 136973df0db0SJames Feist } 1370ecd6a3a2SEd Tanous 1371ecd6a3a2SEd Tanous SetPIDValues(const SetPIDValues&) = delete; 1372ecd6a3a2SEd Tanous SetPIDValues(SetPIDValues&&) = delete; 1373ecd6a3a2SEd Tanous SetPIDValues& operator=(const SetPIDValues&) = delete; 1374ecd6a3a2SEd Tanous SetPIDValues& operator=(SetPIDValues&&) = delete; 1375ecd6a3a2SEd Tanous 137673df0db0SJames Feist void run() 137773df0db0SJames Feist { 137873df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 137973df0db0SJames Feist { 138073df0db0SJames Feist return; 138173df0db0SJames Feist } 138273df0db0SJames Feist 138373df0db0SJames Feist std::shared_ptr<SetPIDValues> self = shared_from_this(); 138473df0db0SJames Feist 138573df0db0SJames Feist // todo(james): might make sense to do a mapper call here if this 138673df0db0SJames Feist // interface gets more traction 138773df0db0SJames Feist crow::connections::systemBus->async_method_call( 138873df0db0SJames Feist [self](const boost::system::error_code ec, 1389914e2d5dSEd Tanous const dbus::utility::ManagedObjectType& mObj) { 139073df0db0SJames Feist if (ec) 139173df0db0SJames Feist { 139273df0db0SJames Feist BMCWEB_LOG_ERROR << "Error communicating to Entity Manager"; 139373df0db0SJames Feist messages::internalError(self->asyncResp->res); 139473df0db0SJames Feist return; 139573df0db0SJames Feist } 1396e69d9de2SJames Feist const std::array<const char*, 3> configurations = { 1397e69d9de2SJames Feist pidConfigurationIface, pidZoneConfigurationIface, 1398e69d9de2SJames Feist stepwiseConfigurationIface}; 1399e69d9de2SJames Feist 140014b0b8d5SJames Feist for (const auto& [path, object] : mObj) 1401e69d9de2SJames Feist { 140214b0b8d5SJames Feist for (const auto& [interface, _] : object) 1403e69d9de2SJames Feist { 1404e69d9de2SJames Feist if (std::find(configurations.begin(), 1405e69d9de2SJames Feist configurations.end(), 1406e69d9de2SJames Feist interface) != configurations.end()) 1407e69d9de2SJames Feist { 140814b0b8d5SJames Feist self->objectCount++; 1409e69d9de2SJames Feist break; 1410e69d9de2SJames Feist } 1411e69d9de2SJames Feist } 1412e69d9de2SJames Feist } 1413914e2d5dSEd Tanous self->managedObj = mObj; 141473df0db0SJames Feist }, 141573df0db0SJames Feist "xyz.openbmc_project.EntityManager", "/", objectManagerIface, 141673df0db0SJames Feist "GetManagedObjects"); 141773df0db0SJames Feist 141873df0db0SJames Feist // at the same time get the profile information 141973df0db0SJames Feist crow::connections::systemBus->async_method_call( 142073df0db0SJames Feist [self](const boost::system::error_code ec, 142173df0db0SJames Feist const crow::openbmc_mapper::GetSubTreeType& subtree) { 142273df0db0SJames Feist if (ec || subtree.empty()) 142373df0db0SJames Feist { 142473df0db0SJames Feist return; 142573df0db0SJames Feist } 142673df0db0SJames Feist if (subtree[0].second.empty()) 142773df0db0SJames Feist { 142873df0db0SJames Feist // invalid mapper response, should never happen 142973df0db0SJames Feist BMCWEB_LOG_ERROR << "SetPIDValues: Mapper Error"; 143073df0db0SJames Feist messages::internalError(self->asyncResp->res); 143173df0db0SJames Feist return; 143273df0db0SJames Feist } 143373df0db0SJames Feist 143473df0db0SJames Feist const std::string& path = subtree[0].first; 143573df0db0SJames Feist const std::string& owner = subtree[0].second[0].first; 143673df0db0SJames Feist crow::connections::systemBus->async_method_call( 143773df0db0SJames Feist [self, path, owner]( 1438cb13a392SEd Tanous const boost::system::error_code ec2, 143973df0db0SJames Feist const boost::container::flat_map< 1440168e20c1SEd Tanous std::string, dbus::utility::DbusVariantType>& r) { 1441cb13a392SEd Tanous if (ec2) 144273df0db0SJames Feist { 14430fda0f12SGeorge Liu BMCWEB_LOG_ERROR 14440fda0f12SGeorge Liu << "SetPIDValues: Can't get thermalModeIface " 144573df0db0SJames Feist << path; 144673df0db0SJames Feist messages::internalError(self->asyncResp->res); 144773df0db0SJames Feist return; 144873df0db0SJames Feist } 1449271584abSEd Tanous const std::string* current = nullptr; 1450271584abSEd Tanous const std::vector<std::string>* supported = nullptr; 14519eb808c1SEd Tanous for (const auto& [key, value] : r) 145273df0db0SJames Feist { 145373df0db0SJames Feist if (key == "Current") 145473df0db0SJames Feist { 145573df0db0SJames Feist current = std::get_if<std::string>(&value); 145673df0db0SJames Feist if (current == nullptr) 145773df0db0SJames Feist { 145873df0db0SJames Feist BMCWEB_LOG_ERROR 14590fda0f12SGeorge Liu << "SetPIDValues: thermal mode iface invalid " 146073df0db0SJames Feist << path; 146173df0db0SJames Feist messages::internalError( 146273df0db0SJames Feist self->asyncResp->res); 146373df0db0SJames Feist return; 146473df0db0SJames Feist } 146573df0db0SJames Feist } 146673df0db0SJames Feist if (key == "Supported") 146773df0db0SJames Feist { 146873df0db0SJames Feist supported = 146973df0db0SJames Feist std::get_if<std::vector<std::string>>( 147073df0db0SJames Feist &value); 147173df0db0SJames Feist if (supported == nullptr) 147273df0db0SJames Feist { 147373df0db0SJames Feist BMCWEB_LOG_ERROR 14740fda0f12SGeorge Liu << "SetPIDValues: thermal mode iface invalid" 147573df0db0SJames Feist << path; 147673df0db0SJames Feist messages::internalError( 147773df0db0SJames Feist self->asyncResp->res); 147873df0db0SJames Feist return; 147973df0db0SJames Feist } 148073df0db0SJames Feist } 148173df0db0SJames Feist } 148273df0db0SJames Feist if (current == nullptr || supported == nullptr) 148373df0db0SJames Feist { 14840fda0f12SGeorge Liu BMCWEB_LOG_ERROR 14850fda0f12SGeorge Liu << "SetPIDValues: thermal mode iface invalid " 148673df0db0SJames Feist << path; 148773df0db0SJames Feist messages::internalError(self->asyncResp->res); 148873df0db0SJames Feist return; 148973df0db0SJames Feist } 149073df0db0SJames Feist self->currentProfile = *current; 149173df0db0SJames Feist self->supportedProfiles = *supported; 149273df0db0SJames Feist self->profileConnection = owner; 149373df0db0SJames Feist self->profilePath = path; 149473df0db0SJames Feist }, 149573df0db0SJames Feist owner, path, "org.freedesktop.DBus.Properties", "GetAll", 149673df0db0SJames Feist thermalModeIface); 14975b4aa86bSJames Feist }, 14985b4aa86bSJames Feist "xyz.openbmc_project.ObjectMapper", 14995b4aa86bSJames Feist "/xyz/openbmc_project/object_mapper", 15005b4aa86bSJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 150173df0db0SJames Feist std::array<const char*, 1>{thermalModeIface}); 150273df0db0SJames Feist } 150324b2fe81SEd Tanous void pidSetDone() 150473df0db0SJames Feist { 150573df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 150673df0db0SJames Feist { 150773df0db0SJames Feist return; 15085b4aa86bSJames Feist } 15098d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> response = asyncResp; 151073df0db0SJames Feist if (profile) 151173df0db0SJames Feist { 151273df0db0SJames Feist if (std::find(supportedProfiles.begin(), supportedProfiles.end(), 151373df0db0SJames Feist *profile) == supportedProfiles.end()) 151473df0db0SJames Feist { 151573df0db0SJames Feist messages::actionParameterUnknown(response->res, "Profile", 151673df0db0SJames Feist *profile); 151773df0db0SJames Feist return; 151873df0db0SJames Feist } 151973df0db0SJames Feist currentProfile = *profile; 152073df0db0SJames Feist crow::connections::systemBus->async_method_call( 152173df0db0SJames Feist [response](const boost::system::error_code ec) { 152273df0db0SJames Feist if (ec) 152373df0db0SJames Feist { 152473df0db0SJames Feist BMCWEB_LOG_ERROR << "Error patching profile" << ec; 152573df0db0SJames Feist messages::internalError(response->res); 152673df0db0SJames Feist } 152773df0db0SJames Feist }, 152873df0db0SJames Feist profileConnection, profilePath, 152973df0db0SJames Feist "org.freedesktop.DBus.Properties", "Set", thermalModeIface, 1530168e20c1SEd Tanous "Current", dbus::utility::DbusVariantType(*profile)); 153173df0db0SJames Feist } 153273df0db0SJames Feist 153373df0db0SJames Feist for (auto& containerPair : configuration) 153473df0db0SJames Feist { 153573df0db0SJames Feist auto& container = containerPair.second; 153673df0db0SJames Feist if (!container) 153773df0db0SJames Feist { 153873df0db0SJames Feist continue; 153973df0db0SJames Feist } 15406ee7f774SJames Feist BMCWEB_LOG_DEBUG << *container; 15416ee7f774SJames Feist 154273df0db0SJames Feist std::string& type = containerPair.first; 154373df0db0SJames Feist 154473df0db0SJames Feist for (nlohmann::json::iterator it = container->begin(); 154517a897dfSManojkiran Eda it != container->end(); ++it) 154673df0db0SJames Feist { 154773df0db0SJames Feist const auto& name = it.key(); 15486ee7f774SJames Feist BMCWEB_LOG_DEBUG << "looking for " << name; 15496ee7f774SJames Feist 155073df0db0SJames Feist auto pathItr = 155173df0db0SJames Feist std::find_if(managedObj.begin(), managedObj.end(), 155273df0db0SJames Feist [&name](const auto& obj) { 155373df0db0SJames Feist return boost::algorithm::ends_with( 155473df0db0SJames Feist obj.first.str, "/" + name); 155573df0db0SJames Feist }); 155673df0db0SJames Feist boost::container::flat_map<std::string, 155773df0db0SJames Feist dbus::utility::DbusVariantType> 155873df0db0SJames Feist output; 155973df0db0SJames Feist 156073df0db0SJames Feist output.reserve(16); // The pid interface length 156173df0db0SJames Feist 156273df0db0SJames Feist // determines if we're patching entity-manager or 156373df0db0SJames Feist // creating a new object 156473df0db0SJames Feist bool createNewObject = (pathItr == managedObj.end()); 15656ee7f774SJames Feist BMCWEB_LOG_DEBUG << "Found = " << !createNewObject; 15666ee7f774SJames Feist 156773df0db0SJames Feist std::string iface; 1568711ac7a9SEd Tanous /* 156973df0db0SJames Feist if (type == "PidControllers" || type == "FanControllers") 157073df0db0SJames Feist { 157173df0db0SJames Feist iface = pidConfigurationIface; 157273df0db0SJames Feist if (!createNewObject && 157373df0db0SJames Feist pathItr->second.find(pidConfigurationIface) == 157473df0db0SJames Feist pathItr->second.end()) 157573df0db0SJames Feist { 157673df0db0SJames Feist createNewObject = true; 157773df0db0SJames Feist } 157873df0db0SJames Feist } 157973df0db0SJames Feist else if (type == "FanZones") 158073df0db0SJames Feist { 158173df0db0SJames Feist iface = pidZoneConfigurationIface; 158273df0db0SJames Feist if (!createNewObject && 158373df0db0SJames Feist pathItr->second.find(pidZoneConfigurationIface) == 158473df0db0SJames Feist pathItr->second.end()) 158573df0db0SJames Feist { 158673df0db0SJames Feist 158773df0db0SJames Feist createNewObject = true; 158873df0db0SJames Feist } 158973df0db0SJames Feist } 159073df0db0SJames Feist else if (type == "StepwiseControllers") 159173df0db0SJames Feist { 159273df0db0SJames Feist iface = stepwiseConfigurationIface; 159373df0db0SJames Feist if (!createNewObject && 159473df0db0SJames Feist pathItr->second.find(stepwiseConfigurationIface) == 159573df0db0SJames Feist pathItr->second.end()) 159673df0db0SJames Feist { 159773df0db0SJames Feist createNewObject = true; 159873df0db0SJames Feist } 1599711ac7a9SEd Tanous }*/ 16006ee7f774SJames Feist 16016ee7f774SJames Feist if (createNewObject && it.value() == nullptr) 16026ee7f774SJames Feist { 16034e0453b1SGunnar Mills // can't delete a non-existent object 1604*ace85d60SEd Tanous messages::propertyValueNotInList(response->res, it.value(), 1605*ace85d60SEd Tanous name); 16066ee7f774SJames Feist continue; 16076ee7f774SJames Feist } 16086ee7f774SJames Feist 16096ee7f774SJames Feist std::string path; 16106ee7f774SJames Feist if (pathItr != managedObj.end()) 16116ee7f774SJames Feist { 16126ee7f774SJames Feist path = pathItr->first.str; 16136ee7f774SJames Feist } 16146ee7f774SJames Feist 161573df0db0SJames Feist BMCWEB_LOG_DEBUG << "Create new = " << createNewObject << "\n"; 1616e69d9de2SJames Feist 1617e69d9de2SJames Feist // arbitrary limit to avoid attacks 1618e69d9de2SJames Feist constexpr const size_t controllerLimit = 500; 161914b0b8d5SJames Feist if (createNewObject && objectCount >= controllerLimit) 1620e69d9de2SJames Feist { 1621e69d9de2SJames Feist messages::resourceExhaustion(response->res, type); 1622e69d9de2SJames Feist continue; 1623e69d9de2SJames Feist } 1624e69d9de2SJames Feist 162573df0db0SJames Feist output["Name"] = boost::replace_all_copy(name, "_", " "); 162673df0db0SJames Feist 162773df0db0SJames Feist std::string chassis; 162873df0db0SJames Feist CreatePIDRet ret = createPidInterface( 16296ee7f774SJames Feist response, type, it, path, managedObj, createNewObject, 16306ee7f774SJames Feist output, chassis, currentProfile); 163173df0db0SJames Feist if (ret == CreatePIDRet::fail) 163273df0db0SJames Feist { 163373df0db0SJames Feist return; 163473df0db0SJames Feist } 16353174e4dfSEd Tanous if (ret == CreatePIDRet::del) 163673df0db0SJames Feist { 163773df0db0SJames Feist continue; 163873df0db0SJames Feist } 163973df0db0SJames Feist 164073df0db0SJames Feist if (!createNewObject) 164173df0db0SJames Feist { 164273df0db0SJames Feist for (const auto& property : output) 164373df0db0SJames Feist { 164473df0db0SJames Feist crow::connections::systemBus->async_method_call( 164573df0db0SJames Feist [response, 164673df0db0SJames Feist propertyName{std::string(property.first)}]( 164773df0db0SJames Feist const boost::system::error_code ec) { 164873df0db0SJames Feist if (ec) 164973df0db0SJames Feist { 165073df0db0SJames Feist BMCWEB_LOG_ERROR << "Error patching " 165173df0db0SJames Feist << propertyName << ": " 165273df0db0SJames Feist << ec; 165373df0db0SJames Feist messages::internalError(response->res); 165473df0db0SJames Feist return; 165573df0db0SJames Feist } 165673df0db0SJames Feist messages::success(response->res); 165773df0db0SJames Feist }, 16586ee7f774SJames Feist "xyz.openbmc_project.EntityManager", path, 165973df0db0SJames Feist "org.freedesktop.DBus.Properties", "Set", iface, 166073df0db0SJames Feist property.first, property.second); 166173df0db0SJames Feist } 166273df0db0SJames Feist } 166373df0db0SJames Feist else 166473df0db0SJames Feist { 166573df0db0SJames Feist if (chassis.empty()) 166673df0db0SJames Feist { 166773df0db0SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config"; 1668*ace85d60SEd Tanous messages::internalError(response->res); 166973df0db0SJames Feist return; 167073df0db0SJames Feist } 167173df0db0SJames Feist 167273df0db0SJames Feist bool foundChassis = false; 167373df0db0SJames Feist for (const auto& obj : managedObj) 167473df0db0SJames Feist { 167573df0db0SJames Feist if (boost::algorithm::ends_with(obj.first.str, chassis)) 167673df0db0SJames Feist { 167773df0db0SJames Feist chassis = obj.first.str; 167873df0db0SJames Feist foundChassis = true; 167973df0db0SJames Feist break; 168073df0db0SJames Feist } 168173df0db0SJames Feist } 168273df0db0SJames Feist if (!foundChassis) 168373df0db0SJames Feist { 168473df0db0SJames Feist BMCWEB_LOG_ERROR << "Failed to find chassis on dbus"; 168573df0db0SJames Feist messages::resourceMissingAtURI( 1686*ace85d60SEd Tanous response->res, 1687*ace85d60SEd Tanous crow::utility::urlFromPieces("redfish", "v1", 1688*ace85d60SEd Tanous "Chassis", chassis)); 168973df0db0SJames Feist return; 169073df0db0SJames Feist } 169173df0db0SJames Feist 169273df0db0SJames Feist crow::connections::systemBus->async_method_call( 169373df0db0SJames Feist [response](const boost::system::error_code ec) { 169473df0db0SJames Feist if (ec) 169573df0db0SJames Feist { 169673df0db0SJames Feist BMCWEB_LOG_ERROR << "Error Adding Pid Object " 169773df0db0SJames Feist << ec; 169873df0db0SJames Feist messages::internalError(response->res); 169973df0db0SJames Feist return; 170073df0db0SJames Feist } 170173df0db0SJames Feist messages::success(response->res); 170273df0db0SJames Feist }, 170373df0db0SJames Feist "xyz.openbmc_project.EntityManager", chassis, 170473df0db0SJames Feist "xyz.openbmc_project.AddObject", "AddObject", output); 170573df0db0SJames Feist } 170673df0db0SJames Feist } 170773df0db0SJames Feist } 170873df0db0SJames Feist } 170924b2fe81SEd Tanous 171024b2fe81SEd Tanous ~SetPIDValues() 171124b2fe81SEd Tanous { 171224b2fe81SEd Tanous try 171324b2fe81SEd Tanous { 171424b2fe81SEd Tanous pidSetDone(); 171524b2fe81SEd Tanous } 171624b2fe81SEd Tanous catch (...) 171724b2fe81SEd Tanous { 171824b2fe81SEd Tanous BMCWEB_LOG_CRITICAL << "pidSetDone threw exception"; 171924b2fe81SEd Tanous } 172024b2fe81SEd Tanous } 172124b2fe81SEd Tanous 17228d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> asyncResp; 172373df0db0SJames Feist std::vector<std::pair<std::string, std::optional<nlohmann::json>>> 172473df0db0SJames Feist configuration; 172573df0db0SJames Feist std::optional<std::string> profile; 172673df0db0SJames Feist dbus::utility::ManagedObjectType managedObj; 172773df0db0SJames Feist std::vector<std::string> supportedProfiles; 172873df0db0SJames Feist std::string currentProfile; 172973df0db0SJames Feist std::string profileConnection; 173073df0db0SJames Feist std::string profilePath; 173114b0b8d5SJames Feist size_t objectCount = 0; 173273df0db0SJames Feist }; 173373df0db0SJames Feist 1734071d8fdfSSunnySrivastava1984 /** 1735071d8fdfSSunnySrivastava1984 * @brief Retrieves BMC manager location data over DBus 1736071d8fdfSSunnySrivastava1984 * 1737071d8fdfSSunnySrivastava1984 * @param[in] aResp Shared pointer for completing asynchronous calls 1738071d8fdfSSunnySrivastava1984 * @param[in] connectionName - service name 1739071d8fdfSSunnySrivastava1984 * @param[in] path - object path 1740071d8fdfSSunnySrivastava1984 * @return none 1741071d8fdfSSunnySrivastava1984 */ 17428d1b46d7Szhanghch05 inline void getLocation(const std::shared_ptr<bmcweb::AsyncResp>& aResp, 1743071d8fdfSSunnySrivastava1984 const std::string& connectionName, 1744071d8fdfSSunnySrivastava1984 const std::string& path) 1745071d8fdfSSunnySrivastava1984 { 1746071d8fdfSSunnySrivastava1984 BMCWEB_LOG_DEBUG << "Get BMC manager Location data."; 1747071d8fdfSSunnySrivastava1984 17481e1e598dSJonathan Doman sdbusplus::asio::getProperty<std::string>( 17491e1e598dSJonathan Doman *crow::connections::systemBus, connectionName, path, 17501e1e598dSJonathan Doman "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode", 1751071d8fdfSSunnySrivastava1984 [aResp](const boost::system::error_code ec, 17521e1e598dSJonathan Doman const std::string& property) { 1753071d8fdfSSunnySrivastava1984 if (ec) 1754071d8fdfSSunnySrivastava1984 { 1755071d8fdfSSunnySrivastava1984 BMCWEB_LOG_DEBUG << "DBUS response error for " 1756071d8fdfSSunnySrivastava1984 "Location"; 1757071d8fdfSSunnySrivastava1984 messages::internalError(aResp->res); 1758071d8fdfSSunnySrivastava1984 return; 1759071d8fdfSSunnySrivastava1984 } 1760071d8fdfSSunnySrivastava1984 1761071d8fdfSSunnySrivastava1984 aResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] = 17621e1e598dSJonathan Doman property; 17631e1e598dSJonathan Doman }); 1764071d8fdfSSunnySrivastava1984 } 17657e860f15SJohn Edward Broadbent // avoid name collision systems.hpp 17667e860f15SJohn Edward Broadbent inline void 17677e860f15SJohn Edward Broadbent managerGetLastResetTime(const std::shared_ptr<bmcweb::AsyncResp>& aResp) 17684bf2b033SGunnar Mills { 17694bf2b033SGunnar Mills BMCWEB_LOG_DEBUG << "Getting Manager Last Reset Time"; 17704bf2b033SGunnar Mills 17711e1e598dSJonathan Doman sdbusplus::asio::getProperty<uint64_t>( 17721e1e598dSJonathan Doman *crow::connections::systemBus, "xyz.openbmc_project.State.BMC", 17731e1e598dSJonathan Doman "/xyz/openbmc_project/state/bmc0", "xyz.openbmc_project.State.BMC", 17741e1e598dSJonathan Doman "LastRebootTime", 17754bf2b033SGunnar Mills [aResp](const boost::system::error_code ec, 17761e1e598dSJonathan Doman const uint64_t lastResetTime) { 17774bf2b033SGunnar Mills if (ec) 17784bf2b033SGunnar Mills { 17794bf2b033SGunnar Mills BMCWEB_LOG_DEBUG << "D-BUS response error " << ec; 17804bf2b033SGunnar Mills return; 17814bf2b033SGunnar Mills } 17824bf2b033SGunnar Mills 17834bf2b033SGunnar Mills // LastRebootTime is epoch time, in milliseconds 17844bf2b033SGunnar Mills // https://github.com/openbmc/phosphor-dbus-interfaces/blob/7f9a128eb9296e926422ddc312c148b625890bb6/xyz/openbmc_project/State/BMC.interface.yaml#L19 17851e1e598dSJonathan Doman uint64_t lastResetTimeStamp = lastResetTime / 1000; 17864bf2b033SGunnar Mills 17874bf2b033SGunnar Mills // Convert to ISO 8601 standard 17884bf2b033SGunnar Mills aResp->res.jsonValue["LastResetTime"] = 17891d8782e7SNan Zhou crow::utility::getDateTimeUint(lastResetTimeStamp); 17901e1e598dSJonathan Doman }); 17914bf2b033SGunnar Mills } 17924bf2b033SGunnar Mills 17934bfefa74SGunnar Mills /** 17944bfefa74SGunnar Mills * @brief Set the running firmware image 17954bfefa74SGunnar Mills * 17964bfefa74SGunnar Mills * @param[i,o] aResp - Async response object 17974bfefa74SGunnar Mills * @param[i] runningFirmwareTarget - Image to make the running image 17984bfefa74SGunnar Mills * 17994bfefa74SGunnar Mills * @return void 18004bfefa74SGunnar Mills */ 18017e860f15SJohn Edward Broadbent inline void 18027e860f15SJohn Edward Broadbent setActiveFirmwareImage(const std::shared_ptr<bmcweb::AsyncResp>& aResp, 1803f23b7296SEd Tanous const std::string& runningFirmwareTarget) 18044bfefa74SGunnar Mills { 18054bfefa74SGunnar Mills // Get the Id from /redfish/v1/UpdateService/FirmwareInventory/<Id> 1806f23b7296SEd Tanous std::string::size_type idPos = runningFirmwareTarget.rfind('/'); 18074bfefa74SGunnar Mills if (idPos == std::string::npos) 18084bfefa74SGunnar Mills { 18094bfefa74SGunnar Mills messages::propertyValueNotInList(aResp->res, runningFirmwareTarget, 18104bfefa74SGunnar Mills "@odata.id"); 18114bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Can't parse firmware ID!"; 18124bfefa74SGunnar Mills return; 18134bfefa74SGunnar Mills } 18144bfefa74SGunnar Mills idPos++; 18154bfefa74SGunnar Mills if (idPos >= runningFirmwareTarget.size()) 18164bfefa74SGunnar Mills { 18174bfefa74SGunnar Mills messages::propertyValueNotInList(aResp->res, runningFirmwareTarget, 18184bfefa74SGunnar Mills "@odata.id"); 18194bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Invalid firmware ID."; 18204bfefa74SGunnar Mills return; 18214bfefa74SGunnar Mills } 18224bfefa74SGunnar Mills std::string firmwareId = runningFirmwareTarget.substr(idPos); 18234bfefa74SGunnar Mills 18244bfefa74SGunnar Mills // Make sure the image is valid before setting priority 18254bfefa74SGunnar Mills crow::connections::systemBus->async_method_call( 1826711ac7a9SEd Tanous [aResp, firmwareId, 1827711ac7a9SEd Tanous runningFirmwareTarget](const boost::system::error_code ec, 1828711ac7a9SEd Tanous dbus::utility::ManagedObjectType& subtree) { 18294bfefa74SGunnar Mills if (ec) 18304bfefa74SGunnar Mills { 18314bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "D-Bus response error getting objects."; 18324bfefa74SGunnar Mills messages::internalError(aResp->res); 18334bfefa74SGunnar Mills return; 18344bfefa74SGunnar Mills } 18354bfefa74SGunnar Mills 183626f6976fSEd Tanous if (subtree.empty()) 18374bfefa74SGunnar Mills { 18384bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Can't find image!"; 18394bfefa74SGunnar Mills messages::internalError(aResp->res); 18404bfefa74SGunnar Mills return; 18414bfefa74SGunnar Mills } 18424bfefa74SGunnar Mills 18434bfefa74SGunnar Mills bool foundImage = false; 18444bfefa74SGunnar Mills for (auto& object : subtree) 18454bfefa74SGunnar Mills { 18464bfefa74SGunnar Mills const std::string& path = 18474bfefa74SGunnar Mills static_cast<const std::string&>(object.first); 1848f23b7296SEd Tanous std::size_t idPos2 = path.rfind('/'); 18494bfefa74SGunnar Mills 18504bfefa74SGunnar Mills if (idPos2 == std::string::npos) 18514bfefa74SGunnar Mills { 18524bfefa74SGunnar Mills continue; 18534bfefa74SGunnar Mills } 18544bfefa74SGunnar Mills 18554bfefa74SGunnar Mills idPos2++; 18564bfefa74SGunnar Mills if (idPos2 >= path.size()) 18574bfefa74SGunnar Mills { 18584bfefa74SGunnar Mills continue; 18594bfefa74SGunnar Mills } 18604bfefa74SGunnar Mills 18614bfefa74SGunnar Mills if (path.substr(idPos2) == firmwareId) 18624bfefa74SGunnar Mills { 18634bfefa74SGunnar Mills foundImage = true; 18644bfefa74SGunnar Mills break; 18654bfefa74SGunnar Mills } 18664bfefa74SGunnar Mills } 18674bfefa74SGunnar Mills 18684bfefa74SGunnar Mills if (!foundImage) 18694bfefa74SGunnar Mills { 18704bfefa74SGunnar Mills messages::propertyValueNotInList( 18714bfefa74SGunnar Mills aResp->res, runningFirmwareTarget, "@odata.id"); 18724bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Invalid firmware ID."; 18734bfefa74SGunnar Mills return; 18744bfefa74SGunnar Mills } 18754bfefa74SGunnar Mills 18767e860f15SJohn Edward Broadbent BMCWEB_LOG_DEBUG 18777e860f15SJohn Edward Broadbent << "Setting firmware version " + firmwareId + " to priority 0."; 18784bfefa74SGunnar Mills 18794bfefa74SGunnar Mills // Only support Immediate 18804bfefa74SGunnar Mills // An addition could be a Redfish Setting like 18814bfefa74SGunnar Mills // ActiveSoftwareImageApplyTime and support OnReset 18824bfefa74SGunnar Mills crow::connections::systemBus->async_method_call( 18834bfefa74SGunnar Mills [aResp](const boost::system::error_code ec) { 18844bfefa74SGunnar Mills if (ec) 18854bfefa74SGunnar Mills { 18864bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "D-Bus response error setting."; 18874bfefa74SGunnar Mills messages::internalError(aResp->res); 18884bfefa74SGunnar Mills return; 18894bfefa74SGunnar Mills } 18904bfefa74SGunnar Mills doBMCGracefulRestart(aResp); 18914bfefa74SGunnar Mills }, 18924bfefa74SGunnar Mills 18934bfefa74SGunnar Mills "xyz.openbmc_project.Software.BMC.Updater", 18944bfefa74SGunnar Mills "/xyz/openbmc_project/software/" + firmwareId, 18954bfefa74SGunnar Mills "org.freedesktop.DBus.Properties", "Set", 18967e860f15SJohn Edward Broadbent "xyz.openbmc_project.Software.RedundancyPriority", "Priority", 1897168e20c1SEd Tanous dbus::utility::DbusVariantType(static_cast<uint8_t>(0))); 18984bfefa74SGunnar Mills }, 18994bfefa74SGunnar Mills "xyz.openbmc_project.Software.BMC.Updater", 19007e860f15SJohn Edward Broadbent "/xyz/openbmc_project/software", "org.freedesktop.DBus.ObjectManager", 19017e860f15SJohn Edward Broadbent "GetManagedObjects"); 19024bfefa74SGunnar Mills } 19034bfefa74SGunnar Mills 19047e860f15SJohn Edward Broadbent inline void setDateTime(std::shared_ptr<bmcweb::AsyncResp> aResp, 19057e860f15SJohn Edward Broadbent std::string datetime) 1906af5d6058SSantosh Puranik { 1907af5d6058SSantosh Puranik BMCWEB_LOG_DEBUG << "Set date time: " << datetime; 1908af5d6058SSantosh Puranik 1909af5d6058SSantosh Puranik std::stringstream stream(datetime); 1910af5d6058SSantosh Puranik // Convert from ISO 8601 to boost local_time 1911af5d6058SSantosh Puranik // (BMC only has time in UTC) 1912af5d6058SSantosh Puranik boost::posix_time::ptime posixTime; 1913af5d6058SSantosh Puranik boost::posix_time::ptime epoch(boost::gregorian::date(1970, 1, 1)); 1914af5d6058SSantosh Puranik // Facet gets deleted with the stringsteam 1915af5d6058SSantosh Puranik auto ifc = std::make_unique<boost::local_time::local_time_input_facet>( 1916af5d6058SSantosh Puranik "%Y-%m-%d %H:%M:%S%F %ZP"); 1917af5d6058SSantosh Puranik stream.imbue(std::locale(stream.getloc(), ifc.release())); 1918af5d6058SSantosh Puranik 19197e860f15SJohn Edward Broadbent boost::local_time::local_date_time ldt(boost::local_time::not_a_date_time); 1920af5d6058SSantosh Puranik 1921af5d6058SSantosh Puranik if (stream >> ldt) 1922af5d6058SSantosh Puranik { 1923af5d6058SSantosh Puranik posixTime = ldt.utc_time(); 1924af5d6058SSantosh Puranik boost::posix_time::time_duration dur = posixTime - epoch; 19257e860f15SJohn Edward Broadbent uint64_t durMicroSecs = static_cast<uint64_t>(dur.total_microseconds()); 1926af5d6058SSantosh Puranik crow::connections::systemBus->async_method_call( 1927af5d6058SSantosh Puranik [aResp{std::move(aResp)}, datetime{std::move(datetime)}]( 1928af5d6058SSantosh Puranik const boost::system::error_code ec) { 1929af5d6058SSantosh Puranik if (ec) 1930af5d6058SSantosh Puranik { 1931af5d6058SSantosh Puranik BMCWEB_LOG_DEBUG << "Failed to set elapsed time. " 1932af5d6058SSantosh Puranik "DBUS response error " 1933af5d6058SSantosh Puranik << ec; 1934af5d6058SSantosh Puranik messages::internalError(aResp->res); 1935af5d6058SSantosh Puranik return; 1936af5d6058SSantosh Puranik } 1937af5d6058SSantosh Puranik aResp->res.jsonValue["DateTime"] = datetime; 1938af5d6058SSantosh Puranik }, 19397e860f15SJohn Edward Broadbent "xyz.openbmc_project.Time.Manager", "/xyz/openbmc_project/time/bmc", 1940af5d6058SSantosh Puranik "org.freedesktop.DBus.Properties", "Set", 1941af5d6058SSantosh Puranik "xyz.openbmc_project.Time.EpochTime", "Elapsed", 1942168e20c1SEd Tanous dbus::utility::DbusVariantType(durMicroSecs)); 1943af5d6058SSantosh Puranik } 1944af5d6058SSantosh Puranik else 1945af5d6058SSantosh Puranik { 19467e860f15SJohn Edward Broadbent messages::propertyValueFormatError(aResp->res, datetime, "DateTime"); 1947af5d6058SSantosh Puranik return; 1948af5d6058SSantosh Puranik } 194983ff9ab6SJames Feist } 19509c310685SBorawski.Lukasz 19517e860f15SJohn Edward Broadbent inline void requestRoutesManager(App& app) 19527e860f15SJohn Edward Broadbent { 19537e860f15SJohn Edward Broadbent std::string uuid = persistent_data::getConfig().systemUuid; 19549c310685SBorawski.Lukasz 19557e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/") 1956ed398213SEd Tanous .privileges(redfish::privileges::getManager) 19577e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)([uuid](const crow::Request&, 19587e860f15SJohn Edward Broadbent const std::shared_ptr< 19597e860f15SJohn Edward Broadbent bmcweb::AsyncResp>& 19607e860f15SJohn Edward Broadbent asyncResp) { 19617e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc"; 19627e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["@odata.type"] = 19637e860f15SJohn Edward Broadbent "#Manager.v1_11_0.Manager"; 19647e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Id"] = "bmc"; 19657e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Name"] = "OpenBmc Manager"; 19667e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Description"] = 19677e860f15SJohn Edward Broadbent "Baseboard Management Controller"; 19687e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["PowerState"] = "On"; 19697e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Status"] = {{"State", "Enabled"}, 19707e860f15SJohn Edward Broadbent {"Health", "OK"}}; 19717e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["ManagerType"] = "BMC"; 19727e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["UUID"] = systemd_utils::getUuid(); 19737e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["ServiceEntryPointUUID"] = uuid; 19747e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Model"] = 19757e860f15SJohn Edward Broadbent "OpenBmc"; // TODO(ed), get model 19767e860f15SJohn Edward Broadbent 19777e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["LogServices"] = { 19787e860f15SJohn Edward Broadbent {"@odata.id", "/redfish/v1/Managers/bmc/LogServices"}}; 19797e860f15SJohn Edward Broadbent 19807e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["NetworkProtocol"] = { 19817e860f15SJohn Edward Broadbent {"@odata.id", "/redfish/v1/Managers/bmc/NetworkProtocol"}}; 19827e860f15SJohn Edward Broadbent 19837e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["EthernetInterfaces"] = { 19847e860f15SJohn Edward Broadbent {"@odata.id", "/redfish/v1/Managers/bmc/EthernetInterfaces"}}; 19857e860f15SJohn Edward Broadbent 19867e860f15SJohn Edward Broadbent #ifdef BMCWEB_ENABLE_VM_NBDPROXY 19877e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["VirtualMedia"] = { 19887e860f15SJohn Edward Broadbent {"@odata.id", "/redfish/v1/Managers/bmc/VirtualMedia"}}; 19897e860f15SJohn Edward Broadbent #endif // BMCWEB_ENABLE_VM_NBDPROXY 19907e860f15SJohn Edward Broadbent 19917e860f15SJohn Edward Broadbent // default oem data 19927e860f15SJohn Edward Broadbent nlohmann::json& oem = asyncResp->res.jsonValue["Oem"]; 19937e860f15SJohn Edward Broadbent nlohmann::json& oemOpenbmc = oem["OpenBmc"]; 19947e860f15SJohn Edward Broadbent oem["@odata.type"] = "#OemManager.Oem"; 19957e860f15SJohn Edward Broadbent oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem"; 19967e860f15SJohn Edward Broadbent oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc"; 19977e860f15SJohn Edward Broadbent oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc"; 19987e860f15SJohn Edward Broadbent oemOpenbmc["Certificates"] = { 19997e860f15SJohn Edward Broadbent {"@odata.id", 20007e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/Truststore/Certificates"}}; 20017e860f15SJohn Edward Broadbent 20027e860f15SJohn Edward Broadbent // Manager.Reset (an action) can be many values, OpenBMC only 20037e860f15SJohn Edward Broadbent // supports BMC reboot. 20047e860f15SJohn Edward Broadbent nlohmann::json& managerReset = 20057e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Actions"]["#Manager.Reset"]; 20067e860f15SJohn Edward Broadbent managerReset["target"] = 20077e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/Actions/Manager.Reset"; 20087e860f15SJohn Edward Broadbent managerReset["@Redfish.ActionInfo"] = 20097e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/ResetActionInfo"; 20107e860f15SJohn Edward Broadbent 20117e860f15SJohn Edward Broadbent // ResetToDefaults (Factory Reset) has values like 20127e860f15SJohn Edward Broadbent // PreserveNetworkAndUsers and PreserveNetwork that aren't supported 20137e860f15SJohn Edward Broadbent // on OpenBMC 20147e860f15SJohn Edward Broadbent nlohmann::json& resetToDefaults = 20157e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Actions"]["#Manager.ResetToDefaults"]; 20167e860f15SJohn Edward Broadbent resetToDefaults["target"] = 20177e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults"; 20187e860f15SJohn Edward Broadbent resetToDefaults["ResetType@Redfish.AllowableValues"] = {"ResetAll"}; 20197e860f15SJohn Edward Broadbent 20207c8c4058STejas Patil std::pair<std::string, std::string> redfishDateTimeOffset = 20217c8c4058STejas Patil crow::utility::getDateTimeOffsetNow(); 20227c8c4058STejas Patil 20237c8c4058STejas Patil asyncResp->res.jsonValue["DateTime"] = redfishDateTimeOffset.first; 20247c8c4058STejas Patil asyncResp->res.jsonValue["DateTimeLocalOffset"] = 20257c8c4058STejas Patil redfishDateTimeOffset.second; 20267e860f15SJohn Edward Broadbent 20270e8ac5e7SGunnar Mills // TODO (Gunnar): Remove these one day since moved to ComputerSystem 20280e8ac5e7SGunnar Mills // Still used by OCP profiles 20290e8ac5e7SGunnar Mills // https://github.com/opencomputeproject/OCP-Profiles/issues/23 20307e860f15SJohn Edward Broadbent // Fill in SerialConsole info 20317e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["SerialConsole"]["ServiceEnabled"] = true; 20327e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["SerialConsole"]["MaxConcurrentSessions"] = 20337e860f15SJohn Edward Broadbent 15; 20347e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["SerialConsole"]["ConnectTypesSupported"] = 20357e860f15SJohn Edward Broadbent {"IPMI", "SSH"}; 20367e860f15SJohn Edward Broadbent #ifdef BMCWEB_ENABLE_KVM 20377e860f15SJohn Edward Broadbent // Fill in GraphicalConsole info 20387e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["GraphicalConsole"]["ServiceEnabled"] = 20397e860f15SJohn Edward Broadbent true; 20407e860f15SJohn Edward Broadbent asyncResp->res 20417e860f15SJohn Edward Broadbent .jsonValue["GraphicalConsole"]["MaxConcurrentSessions"] = 4; 20427e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["GraphicalConsole"] 20437e860f15SJohn Edward Broadbent ["ConnectTypesSupported"] = {"KVMIP"}; 20447e860f15SJohn Edward Broadbent #endif // BMCWEB_ENABLE_KVM 20457e860f15SJohn Edward Broadbent 20467e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Links"]["ManagerForServers@odata.count"] = 20477e860f15SJohn Edward Broadbent 1; 20487e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Links"]["ManagerForServers"] = { 20497e860f15SJohn Edward Broadbent {{"@odata.id", "/redfish/v1/Systems/system"}}}; 20507e860f15SJohn Edward Broadbent 20517e860f15SJohn Edward Broadbent auto health = std::make_shared<HealthPopulate>(asyncResp); 20527e860f15SJohn Edward Broadbent health->isManagersHealth = true; 20537e860f15SJohn Edward Broadbent health->populate(); 20547e860f15SJohn Edward Broadbent 20557e860f15SJohn Edward Broadbent fw_util::populateFirmwareInformation(asyncResp, fw_util::bmcPurpose, 20567e860f15SJohn Edward Broadbent "FirmwareVersion", true); 20577e860f15SJohn Edward Broadbent 20587e860f15SJohn Edward Broadbent managerGetLastResetTime(asyncResp); 20597e860f15SJohn Edward Broadbent 20607e860f15SJohn Edward Broadbent auto pids = std::make_shared<GetPIDValues>(asyncResp); 20617e860f15SJohn Edward Broadbent pids->run(); 20627e860f15SJohn Edward Broadbent 20637e860f15SJohn Edward Broadbent getMainChassisId( 20647e860f15SJohn Edward Broadbent asyncResp, [](const std::string& chassisId, 20657e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& aRsp) { 20667e860f15SJohn Edward Broadbent aRsp->res 20677e860f15SJohn Edward Broadbent .jsonValue["Links"]["ManagerForChassis@odata.count"] = 20687e860f15SJohn Edward Broadbent 1; 20697e860f15SJohn Edward Broadbent aRsp->res.jsonValue["Links"]["ManagerForChassis"] = { 20707e860f15SJohn Edward Broadbent {{"@odata.id", "/redfish/v1/Chassis/" + chassisId}}}; 20717e860f15SJohn Edward Broadbent aRsp->res.jsonValue["Links"]["ManagerInChassis"] = { 20727e860f15SJohn Edward Broadbent {"@odata.id", "/redfish/v1/Chassis/" + chassisId}}; 20737e860f15SJohn Edward Broadbent }); 20747e860f15SJohn Edward Broadbent 20757e860f15SJohn Edward Broadbent static bool started = false; 20767e860f15SJohn Edward Broadbent 20777e860f15SJohn Edward Broadbent if (!started) 20781abe55efSEd Tanous { 20791e1e598dSJonathan Doman sdbusplus::asio::getProperty<double>( 20801e1e598dSJonathan Doman *crow::connections::systemBus, "org.freedesktop.systemd1", 20811e1e598dSJonathan Doman "/org/freedesktop/systemd1", 20821e1e598dSJonathan Doman "org.freedesktop.systemd1.Manager", "Progress", 20837e860f15SJohn Edward Broadbent [asyncResp](const boost::system::error_code ec, 20841e1e598dSJonathan Doman const double& val) { 20857e860f15SJohn Edward Broadbent if (ec) 20861abe55efSEd Tanous { 20877e860f15SJohn Edward Broadbent BMCWEB_LOG_ERROR << "Error while getting progress"; 20887e860f15SJohn Edward Broadbent messages::internalError(asyncResp->res); 20897e860f15SJohn Edward Broadbent return; 20907e860f15SJohn Edward Broadbent } 20911e1e598dSJonathan Doman if (val < 1.0) 20927e860f15SJohn Edward Broadbent { 20937e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Status"]["State"] = 20947e860f15SJohn Edward Broadbent "Starting"; 20957e860f15SJohn Edward Broadbent started = true; 20967e860f15SJohn Edward Broadbent } 20971e1e598dSJonathan Doman }); 20989c310685SBorawski.Lukasz } 20999c310685SBorawski.Lukasz 21007e860f15SJohn Edward Broadbent crow::connections::systemBus->async_method_call( 21017e860f15SJohn Edward Broadbent [asyncResp]( 21027e860f15SJohn Edward Broadbent const boost::system::error_code ec, 21037e860f15SJohn Edward Broadbent const std::vector< 21047e860f15SJohn Edward Broadbent std::pair<std::string, 21057e860f15SJohn Edward Broadbent std::vector<std::pair< 21067e860f15SJohn Edward Broadbent std::string, std::vector<std::string>>>>>& 21077e860f15SJohn Edward Broadbent subtree) { 21087e860f15SJohn Edward Broadbent if (ec) 21091abe55efSEd Tanous { 21107e860f15SJohn Edward Broadbent BMCWEB_LOG_DEBUG 21117e860f15SJohn Edward Broadbent << "D-Bus response error on GetSubTree " << ec; 21127e860f15SJohn Edward Broadbent return; 21137e860f15SJohn Edward Broadbent } 211426f6976fSEd Tanous if (subtree.empty()) 21157e860f15SJohn Edward Broadbent { 21167e860f15SJohn Edward Broadbent BMCWEB_LOG_DEBUG << "Can't find bmc D-Bus object!"; 21177e860f15SJohn Edward Broadbent return; 21187e860f15SJohn Edward Broadbent } 21197e860f15SJohn Edward Broadbent // Assume only 1 bmc D-Bus object 21207e860f15SJohn Edward Broadbent // Throw an error if there is more than 1 21217e860f15SJohn Edward Broadbent if (subtree.size() > 1) 21227e860f15SJohn Edward Broadbent { 21237e860f15SJohn Edward Broadbent BMCWEB_LOG_DEBUG 21247e860f15SJohn Edward Broadbent << "Found more than 1 bmc D-Bus object!"; 21257e860f15SJohn Edward Broadbent messages::internalError(asyncResp->res); 21267e860f15SJohn Edward Broadbent return; 21277e860f15SJohn Edward Broadbent } 21287e860f15SJohn Edward Broadbent 21297e860f15SJohn Edward Broadbent if (subtree[0].first.empty() || 21307e860f15SJohn Edward Broadbent subtree[0].second.size() != 1) 21317e860f15SJohn Edward Broadbent { 21327e860f15SJohn Edward Broadbent BMCWEB_LOG_DEBUG << "Error getting bmc D-Bus object!"; 21337e860f15SJohn Edward Broadbent messages::internalError(asyncResp->res); 21347e860f15SJohn Edward Broadbent return; 21357e860f15SJohn Edward Broadbent } 21367e860f15SJohn Edward Broadbent 21377e860f15SJohn Edward Broadbent const std::string& path = subtree[0].first; 21387e860f15SJohn Edward Broadbent const std::string& connectionName = 21397e860f15SJohn Edward Broadbent subtree[0].second[0].first; 21407e860f15SJohn Edward Broadbent 21417e860f15SJohn Edward Broadbent for (const auto& interfaceName : 21427e860f15SJohn Edward Broadbent subtree[0].second[0].second) 21437e860f15SJohn Edward Broadbent { 21447e860f15SJohn Edward Broadbent if (interfaceName == 21457e860f15SJohn Edward Broadbent "xyz.openbmc_project.Inventory.Decorator.Asset") 21467e860f15SJohn Edward Broadbent { 21477e860f15SJohn Edward Broadbent crow::connections::systemBus->async_method_call( 21487e860f15SJohn Edward Broadbent [asyncResp]( 21497e860f15SJohn Edward Broadbent const boost::system::error_code ec, 2150168e20c1SEd Tanous const std::vector<std::pair< 2151168e20c1SEd Tanous std::string, 2152168e20c1SEd Tanous dbus::utility::DbusVariantType>>& 21537e860f15SJohn Edward Broadbent propertiesList) { 21547e860f15SJohn Edward Broadbent if (ec) 21557e860f15SJohn Edward Broadbent { 21567e860f15SJohn Edward Broadbent BMCWEB_LOG_DEBUG 21577e860f15SJohn Edward Broadbent << "Can't get bmc asset!"; 21587e860f15SJohn Edward Broadbent return; 21597e860f15SJohn Edward Broadbent } 21607e860f15SJohn Edward Broadbent for (const std::pair< 21617e860f15SJohn Edward Broadbent std::string, 2162168e20c1SEd Tanous dbus::utility::DbusVariantType>& 21637e860f15SJohn Edward Broadbent property : propertiesList) 21647e860f15SJohn Edward Broadbent { 21657e860f15SJohn Edward Broadbent const std::string& propertyName = 21667e860f15SJohn Edward Broadbent property.first; 21677e860f15SJohn Edward Broadbent 21687e860f15SJohn Edward Broadbent if ((propertyName == "PartNumber") || 21697e860f15SJohn Edward Broadbent (propertyName == "SerialNumber") || 21707e860f15SJohn Edward Broadbent (propertyName == "Manufacturer") || 21717e860f15SJohn Edward Broadbent (propertyName == "Model") || 21727e860f15SJohn Edward Broadbent (propertyName == "SparePartNumber")) 21737e860f15SJohn Edward Broadbent { 21747e860f15SJohn Edward Broadbent const std::string* value = 21757e860f15SJohn Edward Broadbent std::get_if<std::string>( 21767e860f15SJohn Edward Broadbent &property.second); 21777e860f15SJohn Edward Broadbent if (value == nullptr) 21787e860f15SJohn Edward Broadbent { 21797e860f15SJohn Edward Broadbent // illegal property 21807e860f15SJohn Edward Broadbent messages::internalError( 21817e860f15SJohn Edward Broadbent asyncResp->res); 21827e860f15SJohn Edward Broadbent return; 21837e860f15SJohn Edward Broadbent } 21847e860f15SJohn Edward Broadbent asyncResp->res 21857e860f15SJohn Edward Broadbent .jsonValue[propertyName] = 21867e860f15SJohn Edward Broadbent *value; 21877e860f15SJohn Edward Broadbent } 21887e860f15SJohn Edward Broadbent } 21897e860f15SJohn Edward Broadbent }, 21907e860f15SJohn Edward Broadbent connectionName, path, 21917e860f15SJohn Edward Broadbent "org.freedesktop.DBus.Properties", "GetAll", 21920fda0f12SGeorge Liu "xyz.openbmc_project.Inventory.Decorator.Asset"); 21937e860f15SJohn Edward Broadbent } 21940fda0f12SGeorge Liu else if ( 21950fda0f12SGeorge Liu 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) 22127e860f15SJohn Edward Broadbent .methods( 22137e860f15SJohn Edward Broadbent boost::beast::http::verb:: 22147e860f15SJohn Edward Broadbent patch)([](const crow::Request& req, 22157e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 22167e860f15SJohn Edward Broadbent std::optional<nlohmann::json> oem; 22177e860f15SJohn Edward Broadbent std::optional<nlohmann::json> links; 22187e860f15SJohn Edward Broadbent std::optional<std::string> datetime; 22197e860f15SJohn Edward Broadbent 222015ed6780SWilly Tu if (!json_util::readJsonPatch(req, asyncResp->res, "Oem", oem, 22217e860f15SJohn Edward Broadbent "DateTime", datetime, "Links", links)) 22227e860f15SJohn Edward Broadbent { 22237e860f15SJohn Edward Broadbent return; 22247e860f15SJohn Edward Broadbent } 22257e860f15SJohn Edward Broadbent 22267e860f15SJohn Edward Broadbent if (oem) 22277e860f15SJohn Edward Broadbent { 22287e860f15SJohn Edward Broadbent std::optional<nlohmann::json> openbmc; 22297e860f15SJohn Edward Broadbent if (!redfish::json_util::readJson(*oem, asyncResp->res, 22307e860f15SJohn Edward Broadbent "OpenBmc", openbmc)) 22317e860f15SJohn Edward Broadbent { 22327e860f15SJohn Edward Broadbent BMCWEB_LOG_ERROR 22337e860f15SJohn Edward Broadbent << "Illegal Property " 22347e860f15SJohn Edward Broadbent << oem->dump(2, ' ', true, 22357e860f15SJohn Edward Broadbent nlohmann::json::error_handler_t::replace); 22367e860f15SJohn Edward Broadbent return; 22377e860f15SJohn Edward Broadbent } 22387e860f15SJohn Edward Broadbent if (openbmc) 22397e860f15SJohn Edward Broadbent { 22407e860f15SJohn Edward Broadbent std::optional<nlohmann::json> fan; 22417e860f15SJohn Edward Broadbent if (!redfish::json_util::readJson(*openbmc, asyncResp->res, 22427e860f15SJohn Edward Broadbent "Fan", fan)) 22437e860f15SJohn Edward Broadbent { 22447e860f15SJohn Edward Broadbent BMCWEB_LOG_ERROR 22457e860f15SJohn Edward Broadbent << "Illegal Property " 22467e860f15SJohn Edward Broadbent << openbmc->dump( 22477e860f15SJohn Edward Broadbent 2, ' ', true, 22487e860f15SJohn Edward Broadbent nlohmann::json::error_handler_t::replace); 22497e860f15SJohn Edward Broadbent return; 22507e860f15SJohn Edward Broadbent } 22517e860f15SJohn Edward Broadbent if (fan) 22527e860f15SJohn Edward Broadbent { 22537e860f15SJohn Edward Broadbent auto pid = 22547e860f15SJohn Edward Broadbent std::make_shared<SetPIDValues>(asyncResp, *fan); 22557e860f15SJohn Edward Broadbent pid->run(); 22567e860f15SJohn Edward Broadbent } 22577e860f15SJohn Edward Broadbent } 22587e860f15SJohn Edward Broadbent } 22597e860f15SJohn Edward Broadbent if (links) 22607e860f15SJohn Edward Broadbent { 22617e860f15SJohn Edward Broadbent std::optional<nlohmann::json> activeSoftwareImage; 22627e860f15SJohn Edward Broadbent if (!redfish::json_util::readJson(*links, asyncResp->res, 22637e860f15SJohn Edward Broadbent "ActiveSoftwareImage", 22647e860f15SJohn Edward Broadbent activeSoftwareImage)) 22657e860f15SJohn Edward Broadbent { 22667e860f15SJohn Edward Broadbent return; 22677e860f15SJohn Edward Broadbent } 22687e860f15SJohn Edward Broadbent if (activeSoftwareImage) 22697e860f15SJohn Edward Broadbent { 22707e860f15SJohn Edward Broadbent std::optional<std::string> odataId; 22717e860f15SJohn Edward Broadbent if (!json_util::readJson(*activeSoftwareImage, 22727e860f15SJohn Edward Broadbent asyncResp->res, "@odata.id", 22737e860f15SJohn Edward Broadbent odataId)) 22747e860f15SJohn Edward Broadbent { 22757e860f15SJohn Edward Broadbent return; 22767e860f15SJohn Edward Broadbent } 22777e860f15SJohn Edward Broadbent 22787e860f15SJohn Edward Broadbent if (odataId) 22797e860f15SJohn Edward Broadbent { 22807e860f15SJohn Edward Broadbent setActiveFirmwareImage(asyncResp, *odataId); 22817e860f15SJohn Edward Broadbent } 22827e860f15SJohn Edward Broadbent } 22837e860f15SJohn Edward Broadbent } 22847e860f15SJohn Edward Broadbent if (datetime) 22857e860f15SJohn Edward Broadbent { 22867e860f15SJohn Edward Broadbent setDateTime(asyncResp, std::move(*datetime)); 22877e860f15SJohn Edward Broadbent } 22887e860f15SJohn Edward Broadbent }); 22897e860f15SJohn Edward Broadbent } 22907e860f15SJohn Edward Broadbent 22917e860f15SJohn Edward Broadbent inline void requestRoutesManagerCollection(App& app) 22927e860f15SJohn Edward Broadbent { 22937e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/") 2294ed398213SEd Tanous .privileges(redfish::privileges::getManagerCollection) 22957e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)( 22967e860f15SJohn Edward Broadbent [](const crow::Request&, 22977e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 229883ff9ab6SJames Feist // Collections don't include the static data added by SubRoute 229983ff9ab6SJames Feist // because it has a duplicate entry for members 23008d1b46d7Szhanghch05 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers"; 23018d1b46d7Szhanghch05 asyncResp->res.jsonValue["@odata.type"] = 23028d1b46d7Szhanghch05 "#ManagerCollection.ManagerCollection"; 23038d1b46d7Szhanghch05 asyncResp->res.jsonValue["Name"] = "Manager Collection"; 23048d1b46d7Szhanghch05 asyncResp->res.jsonValue["Members@odata.count"] = 1; 23058d1b46d7Szhanghch05 asyncResp->res.jsonValue["Members"] = { 23065b4aa86bSJames Feist {{"@odata.id", "/redfish/v1/Managers/bmc"}}}; 23077e860f15SJohn Edward Broadbent }); 23089c310685SBorawski.Lukasz } 23099c310685SBorawski.Lukasz } // namespace redfish 2310