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" 199c310685SBorawski.Lukasz #include "node.hpp" 20c5d03ff4SJennifer Lee #include "redfish_util.hpp" 219c310685SBorawski.Lukasz 225b4aa86bSJames Feist #include <boost/algorithm/string/replace.hpp> 23af5d6058SSantosh Puranik #include <boost/date_time.hpp> 245b4aa86bSJames Feist #include <dbus_utility.hpp> 25e90c5052SAndrew Geissler #include <utils/fw_utils.hpp> 267bffdb7eSBernard Wong #include <utils/systemd_utils.hpp> 271214b7e7SGunnar Mills 284bfefa74SGunnar Mills #include <cstdint> 291214b7e7SGunnar Mills #include <memory> 301214b7e7SGunnar Mills #include <sstream> 31abf2add6SEd Tanous #include <variant> 325b4aa86bSJames Feist 331abe55efSEd Tanous namespace redfish 341abe55efSEd Tanous { 35ed5befbdSJennifer Lee 36ed5befbdSJennifer Lee /** 372a5c4407SGunnar Mills * Function reboots the BMC. 382a5c4407SGunnar Mills * 392a5c4407SGunnar Mills * @param[in] asyncResp - Shared pointer for completing asynchronous calls 40ed5befbdSJennifer Lee */ 41*8d1b46d7Szhanghch05 inline void 42*8d1b46d7Szhanghch05 doBMCGracefulRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 43ed5befbdSJennifer Lee { 44ed5befbdSJennifer Lee const char* processName = "xyz.openbmc_project.State.BMC"; 45ed5befbdSJennifer Lee const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 46ed5befbdSJennifer Lee const char* interfaceName = "xyz.openbmc_project.State.BMC"; 47ed5befbdSJennifer Lee const std::string& propertyValue = 48ed5befbdSJennifer Lee "xyz.openbmc_project.State.BMC.Transition.Reboot"; 49ed5befbdSJennifer Lee const char* destProperty = "RequestedBMCTransition"; 50ed5befbdSJennifer Lee 51ed5befbdSJennifer Lee // Create the D-Bus variant for D-Bus call. 52ed5befbdSJennifer Lee VariantType dbusPropertyValue(propertyValue); 53ed5befbdSJennifer Lee 54ed5befbdSJennifer Lee crow::connections::systemBus->async_method_call( 55ed5befbdSJennifer Lee [asyncResp](const boost::system::error_code ec) { 56ed5befbdSJennifer Lee // Use "Set" method to set the property value. 57ed5befbdSJennifer Lee if (ec) 58ed5befbdSJennifer Lee { 592a5c4407SGunnar Mills BMCWEB_LOG_DEBUG << "[Set] Bad D-Bus request error: " << ec; 60ed5befbdSJennifer Lee messages::internalError(asyncResp->res); 61ed5befbdSJennifer Lee return; 62ed5befbdSJennifer Lee } 63ed5befbdSJennifer Lee 64ed5befbdSJennifer Lee messages::success(asyncResp->res); 65ed5befbdSJennifer Lee }, 66ed5befbdSJennifer Lee processName, objectPath, "org.freedesktop.DBus.Properties", "Set", 67ed5befbdSJennifer Lee interfaceName, destProperty, dbusPropertyValue); 68ed5befbdSJennifer Lee } 692a5c4407SGunnar Mills 70*8d1b46d7Szhanghch05 inline void 71*8d1b46d7Szhanghch05 doBMCForceRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 72f92af389SJayaprakash Mutyala { 73f92af389SJayaprakash Mutyala const char* processName = "xyz.openbmc_project.State.BMC"; 74f92af389SJayaprakash Mutyala const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 75f92af389SJayaprakash Mutyala const char* interfaceName = "xyz.openbmc_project.State.BMC"; 76f92af389SJayaprakash Mutyala const std::string& propertyValue = 77f92af389SJayaprakash Mutyala "xyz.openbmc_project.State.BMC.Transition.HardReboot"; 78f92af389SJayaprakash Mutyala const char* destProperty = "RequestedBMCTransition"; 79f92af389SJayaprakash Mutyala 80f92af389SJayaprakash Mutyala // Create the D-Bus variant for D-Bus call. 81f92af389SJayaprakash Mutyala VariantType dbusPropertyValue(propertyValue); 82f92af389SJayaprakash Mutyala 83f92af389SJayaprakash Mutyala crow::connections::systemBus->async_method_call( 84f92af389SJayaprakash Mutyala [asyncResp](const boost::system::error_code ec) { 85f92af389SJayaprakash Mutyala // Use "Set" method to set the property value. 86f92af389SJayaprakash Mutyala if (ec) 87f92af389SJayaprakash Mutyala { 88f92af389SJayaprakash Mutyala BMCWEB_LOG_DEBUG << "[Set] Bad D-Bus request error: " << ec; 89f92af389SJayaprakash Mutyala messages::internalError(asyncResp->res); 90f92af389SJayaprakash Mutyala return; 91f92af389SJayaprakash Mutyala } 92f92af389SJayaprakash Mutyala 93f92af389SJayaprakash Mutyala messages::success(asyncResp->res); 94f92af389SJayaprakash Mutyala }, 95f92af389SJayaprakash Mutyala processName, objectPath, "org.freedesktop.DBus.Properties", "Set", 96f92af389SJayaprakash Mutyala interfaceName, destProperty, dbusPropertyValue); 97f92af389SJayaprakash Mutyala } 98f92af389SJayaprakash Mutyala 992a5c4407SGunnar Mills /** 1002a5c4407SGunnar Mills * ManagerResetAction class supports the POST method for the Reset (reboot) 1012a5c4407SGunnar Mills * action. 1022a5c4407SGunnar Mills */ 1032a5c4407SGunnar Mills class ManagerResetAction : public Node 1042a5c4407SGunnar Mills { 1052a5c4407SGunnar Mills public: 10652cc112dSEd Tanous ManagerResetAction(App& app) : 1072a5c4407SGunnar Mills Node(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/") 1082a5c4407SGunnar Mills { 1092a5c4407SGunnar Mills entityPrivileges = { 1102a5c4407SGunnar Mills {boost::beast::http::verb::post, {{"ConfigureManager"}}}}; 1112a5c4407SGunnar Mills } 1122a5c4407SGunnar Mills 1132a5c4407SGunnar Mills private: 1142a5c4407SGunnar Mills /** 1152a5c4407SGunnar Mills * Function handles POST method request. 1162a5c4407SGunnar Mills * Analyzes POST body before sending Reset (Reboot) request data to D-Bus. 117f92af389SJayaprakash Mutyala * OpenBMC supports ResetType "GracefulRestart" and "ForceRestart". 1182a5c4407SGunnar Mills */ 119*8d1b46d7Szhanghch05 void doPost(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 120*8d1b46d7Szhanghch05 const crow::Request& req, 121cb13a392SEd Tanous const std::vector<std::string>&) override 1222a5c4407SGunnar Mills { 1232a5c4407SGunnar Mills BMCWEB_LOG_DEBUG << "Post Manager Reset."; 1242a5c4407SGunnar Mills 1252a5c4407SGunnar Mills std::string resetType; 1262a5c4407SGunnar Mills 1272a5c4407SGunnar Mills if (!json_util::readJson(req, asyncResp->res, "ResetType", resetType)) 1282a5c4407SGunnar Mills { 1292a5c4407SGunnar Mills return; 1302a5c4407SGunnar Mills } 1312a5c4407SGunnar Mills 132f92af389SJayaprakash Mutyala if (resetType == "GracefulRestart") 133f92af389SJayaprakash Mutyala { 134f92af389SJayaprakash Mutyala BMCWEB_LOG_DEBUG << "Proceeding with " << resetType; 135f92af389SJayaprakash Mutyala doBMCGracefulRestart(asyncResp); 136f92af389SJayaprakash Mutyala return; 137f92af389SJayaprakash Mutyala } 1383174e4dfSEd Tanous if (resetType == "ForceRestart") 139f92af389SJayaprakash Mutyala { 140f92af389SJayaprakash Mutyala BMCWEB_LOG_DEBUG << "Proceeding with " << resetType; 141f92af389SJayaprakash Mutyala doBMCForceRestart(asyncResp); 142f92af389SJayaprakash Mutyala return; 143f92af389SJayaprakash Mutyala } 1442a5c4407SGunnar Mills BMCWEB_LOG_DEBUG << "Invalid property value for ResetType: " 1452a5c4407SGunnar Mills << resetType; 1462a5c4407SGunnar Mills messages::actionParameterNotSupported(asyncResp->res, resetType, 1472a5c4407SGunnar Mills "ResetType"); 1482a5c4407SGunnar Mills 1492a5c4407SGunnar Mills return; 1502a5c4407SGunnar Mills } 151ed5befbdSJennifer Lee }; 152ed5befbdSJennifer Lee 1533e40fc74SGunnar Mills /** 1543e40fc74SGunnar Mills * ManagerResetToDefaultsAction class supports POST method for factory reset 1553e40fc74SGunnar Mills * action. 1563e40fc74SGunnar Mills */ 1573e40fc74SGunnar Mills class ManagerResetToDefaultsAction : public Node 1583e40fc74SGunnar Mills { 1593e40fc74SGunnar Mills public: 16052cc112dSEd Tanous ManagerResetToDefaultsAction(App& app) : 1613e40fc74SGunnar Mills Node(app, "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults/") 1623e40fc74SGunnar Mills { 1633e40fc74SGunnar Mills entityPrivileges = { 1643e40fc74SGunnar Mills {boost::beast::http::verb::post, {{"ConfigureManager"}}}}; 1653e40fc74SGunnar Mills } 1663e40fc74SGunnar Mills 1673e40fc74SGunnar Mills private: 1683e40fc74SGunnar Mills /** 1693e40fc74SGunnar Mills * Function handles ResetToDefaults POST method request. 1703e40fc74SGunnar Mills * 1713e40fc74SGunnar Mills * Analyzes POST body message and factory resets BMC by calling 1723e40fc74SGunnar Mills * BMC code updater factory reset followed by a BMC reboot. 1733e40fc74SGunnar Mills * 1743e40fc74SGunnar Mills * BMC code updater factory reset wipes the whole BMC read-write 1753e40fc74SGunnar Mills * filesystem which includes things like the network settings. 1763e40fc74SGunnar Mills * 1773e40fc74SGunnar Mills * OpenBMC only supports ResetToDefaultsType "ResetAll". 1783e40fc74SGunnar Mills */ 179*8d1b46d7Szhanghch05 void doPost(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 180*8d1b46d7Szhanghch05 const crow::Request& req, 181cb13a392SEd Tanous const std::vector<std::string>&) override 1823e40fc74SGunnar Mills { 1833e40fc74SGunnar Mills BMCWEB_LOG_DEBUG << "Post ResetToDefaults."; 1843e40fc74SGunnar Mills 1853e40fc74SGunnar Mills std::string resetType; 1863e40fc74SGunnar Mills 1873e40fc74SGunnar Mills if (!json_util::readJson(req, asyncResp->res, "ResetToDefaultsType", 1883e40fc74SGunnar Mills resetType)) 1893e40fc74SGunnar Mills { 1903e40fc74SGunnar Mills BMCWEB_LOG_DEBUG << "Missing property ResetToDefaultsType."; 1913e40fc74SGunnar Mills 1923e40fc74SGunnar Mills messages::actionParameterMissing(asyncResp->res, "ResetToDefaults", 1933e40fc74SGunnar Mills "ResetToDefaultsType"); 1943e40fc74SGunnar Mills return; 1953e40fc74SGunnar Mills } 1963e40fc74SGunnar Mills 1973e40fc74SGunnar Mills if (resetType != "ResetAll") 1983e40fc74SGunnar Mills { 1993e40fc74SGunnar Mills BMCWEB_LOG_DEBUG << "Invalid property value for " 2003e40fc74SGunnar Mills "ResetToDefaultsType: " 2013e40fc74SGunnar Mills << resetType; 2023e40fc74SGunnar Mills messages::actionParameterNotSupported(asyncResp->res, resetType, 2033e40fc74SGunnar Mills "ResetToDefaultsType"); 2043e40fc74SGunnar Mills return; 2053e40fc74SGunnar Mills } 2063e40fc74SGunnar Mills 2073e40fc74SGunnar Mills crow::connections::systemBus->async_method_call( 2083e40fc74SGunnar Mills [asyncResp](const boost::system::error_code ec) { 2093e40fc74SGunnar Mills if (ec) 2103e40fc74SGunnar Mills { 2113e40fc74SGunnar Mills BMCWEB_LOG_DEBUG << "Failed to ResetToDefaults: " << ec; 2123e40fc74SGunnar Mills messages::internalError(asyncResp->res); 2133e40fc74SGunnar Mills return; 2143e40fc74SGunnar Mills } 2153e40fc74SGunnar Mills // Factory Reset doesn't actually happen until a reboot 2163e40fc74SGunnar Mills // Can't erase what the BMC is running on 2173e40fc74SGunnar Mills doBMCGracefulRestart(asyncResp); 2183e40fc74SGunnar Mills }, 2193e40fc74SGunnar Mills "xyz.openbmc_project.Software.BMC.Updater", 2203e40fc74SGunnar Mills "/xyz/openbmc_project/software", 2213e40fc74SGunnar Mills "xyz.openbmc_project.Common.FactoryReset", "Reset"); 2223e40fc74SGunnar Mills } 2233e40fc74SGunnar Mills }; 2243e40fc74SGunnar Mills 2251cb1a9e6SAppaRao Puli /** 2261cb1a9e6SAppaRao Puli * ManagerResetActionInfo derived class for delivering Manager 2271cb1a9e6SAppaRao Puli * ResetType AllowableValues using ResetInfo schema. 2281cb1a9e6SAppaRao Puli */ 2291cb1a9e6SAppaRao Puli class ManagerResetActionInfo : public Node 2301cb1a9e6SAppaRao Puli { 2311cb1a9e6SAppaRao Puli public: 2321cb1a9e6SAppaRao Puli /* 2331cb1a9e6SAppaRao Puli * Default Constructor 2341cb1a9e6SAppaRao Puli */ 23552cc112dSEd Tanous ManagerResetActionInfo(App& app) : 2361cb1a9e6SAppaRao Puli Node(app, "/redfish/v1/Managers/bmc/ResetActionInfo/") 2371cb1a9e6SAppaRao Puli { 2381cb1a9e6SAppaRao Puli entityPrivileges = { 2391cb1a9e6SAppaRao Puli {boost::beast::http::verb::get, {{"Login"}}}, 2401cb1a9e6SAppaRao Puli {boost::beast::http::verb::head, {{"Login"}}}, 2411cb1a9e6SAppaRao Puli {boost::beast::http::verb::patch, {{"ConfigureComponents"}}}, 2421cb1a9e6SAppaRao Puli {boost::beast::http::verb::put, {{"ConfigureComponents"}}}, 2431cb1a9e6SAppaRao Puli {boost::beast::http::verb::delete_, {{"ConfigureComponents"}}}, 2441cb1a9e6SAppaRao Puli {boost::beast::http::verb::post, {{"ConfigureComponents"}}}}; 2451cb1a9e6SAppaRao Puli } 2461cb1a9e6SAppaRao Puli 2471cb1a9e6SAppaRao Puli private: 2481cb1a9e6SAppaRao Puli /** 2491cb1a9e6SAppaRao Puli * Functions triggers appropriate requests on DBus 2501cb1a9e6SAppaRao Puli */ 251*8d1b46d7Szhanghch05 void doGet(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 252*8d1b46d7Szhanghch05 const crow::Request&, const std::vector<std::string>&) override 2531cb1a9e6SAppaRao Puli { 254*8d1b46d7Szhanghch05 asyncResp->res.jsonValue = { 2551cb1a9e6SAppaRao Puli {"@odata.type", "#ActionInfo.v1_1_2.ActionInfo"}, 2561cb1a9e6SAppaRao Puli {"@odata.id", "/redfish/v1/Managers/bmc/ResetActionInfo"}, 2571cb1a9e6SAppaRao Puli {"Name", "Reset Action Info"}, 2581cb1a9e6SAppaRao Puli {"Id", "ResetActionInfo"}, 2591cb1a9e6SAppaRao Puli {"Parameters", 2601cb1a9e6SAppaRao Puli {{{"Name", "ResetType"}, 2611cb1a9e6SAppaRao Puli {"Required", true}, 2621cb1a9e6SAppaRao Puli {"DataType", "String"}, 263f92af389SJayaprakash Mutyala {"AllowableValues", {"GracefulRestart", "ForceRestart"}}}}}}; 2641cb1a9e6SAppaRao Puli } 2651cb1a9e6SAppaRao Puli }; 2661cb1a9e6SAppaRao Puli 2675b4aa86bSJames Feist static constexpr const char* objectManagerIface = 2685b4aa86bSJames Feist "org.freedesktop.DBus.ObjectManager"; 2695b4aa86bSJames Feist static constexpr const char* pidConfigurationIface = 2705b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid"; 2715b4aa86bSJames Feist static constexpr const char* pidZoneConfigurationIface = 2725b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid.Zone"; 273b7a08d04SJames Feist static constexpr const char* stepwiseConfigurationIface = 274b7a08d04SJames Feist "xyz.openbmc_project.Configuration.Stepwise"; 27573df0db0SJames Feist static constexpr const char* thermalModeIface = 27673df0db0SJames Feist "xyz.openbmc_project.Control.ThermalMode"; 2779c310685SBorawski.Lukasz 278*8d1b46d7Szhanghch05 inline void 279*8d1b46d7Szhanghch05 asyncPopulatePid(const std::string& connection, const std::string& path, 28073df0db0SJames Feist const std::string& currentProfile, 28173df0db0SJames Feist const std::vector<std::string>& supportedProfiles, 282*8d1b46d7Szhanghch05 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 2835b4aa86bSJames Feist { 2845b4aa86bSJames Feist 2855b4aa86bSJames Feist crow::connections::systemBus->async_method_call( 28673df0db0SJames Feist [asyncResp, currentProfile, supportedProfiles]( 28773df0db0SJames Feist const boost::system::error_code ec, 2885b4aa86bSJames Feist const dbus::utility::ManagedObjectType& managedObj) { 2895b4aa86bSJames Feist if (ec) 2905b4aa86bSJames Feist { 2915b4aa86bSJames Feist BMCWEB_LOG_ERROR << ec; 2925b4aa86bSJames Feist asyncResp->res.jsonValue.clear(); 293f12894f8SJason M. Bills messages::internalError(asyncResp->res); 2945b4aa86bSJames Feist return; 2955b4aa86bSJames Feist } 2965b4aa86bSJames Feist nlohmann::json& configRoot = 2975b4aa86bSJames Feist asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"]; 2985b4aa86bSJames Feist nlohmann::json& fans = configRoot["FanControllers"]; 2995b4aa86bSJames Feist fans["@odata.type"] = "#OemManager.FanControllers"; 3005b4aa86bSJames Feist fans["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/" 3015b4aa86bSJames Feist "Fan/FanControllers"; 3025b4aa86bSJames Feist 3035b4aa86bSJames Feist nlohmann::json& pids = configRoot["PidControllers"]; 3045b4aa86bSJames Feist pids["@odata.type"] = "#OemManager.PidControllers"; 3055b4aa86bSJames Feist pids["@odata.id"] = 3065b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers"; 3075b4aa86bSJames Feist 308b7a08d04SJames Feist nlohmann::json& stepwise = configRoot["StepwiseControllers"]; 309b7a08d04SJames Feist stepwise["@odata.type"] = "#OemManager.StepwiseControllers"; 310b7a08d04SJames Feist stepwise["@odata.id"] = 311b7a08d04SJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers"; 312b7a08d04SJames Feist 3135b4aa86bSJames Feist nlohmann::json& zones = configRoot["FanZones"]; 3145b4aa86bSJames Feist zones["@odata.id"] = 3155b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones"; 3165b4aa86bSJames Feist zones["@odata.type"] = "#OemManager.FanZones"; 3175b4aa86bSJames Feist configRoot["@odata.id"] = 3185b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan"; 3195b4aa86bSJames Feist configRoot["@odata.type"] = "#OemManager.Fan"; 32073df0db0SJames Feist configRoot["Profile@Redfish.AllowableValues"] = supportedProfiles; 32173df0db0SJames Feist 32273df0db0SJames Feist if (!currentProfile.empty()) 32373df0db0SJames Feist { 32473df0db0SJames Feist configRoot["Profile"] = currentProfile; 32573df0db0SJames Feist } 32673df0db0SJames Feist BMCWEB_LOG_ERROR << "profile = " << currentProfile << " !"; 3275b4aa86bSJames Feist 3285b4aa86bSJames Feist for (const auto& pathPair : managedObj) 3295b4aa86bSJames Feist { 3305b4aa86bSJames Feist for (const auto& intfPair : pathPair.second) 3315b4aa86bSJames Feist { 3325b4aa86bSJames Feist if (intfPair.first != pidConfigurationIface && 333b7a08d04SJames Feist intfPair.first != pidZoneConfigurationIface && 334b7a08d04SJames Feist intfPair.first != stepwiseConfigurationIface) 3355b4aa86bSJames Feist { 3365b4aa86bSJames Feist continue; 3375b4aa86bSJames Feist } 3385b4aa86bSJames Feist auto findName = intfPair.second.find("Name"); 3395b4aa86bSJames Feist if (findName == intfPair.second.end()) 3405b4aa86bSJames Feist { 3415b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Field missing Name"; 342a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 3435b4aa86bSJames Feist return; 3445b4aa86bSJames Feist } 34573df0db0SJames Feist 3465b4aa86bSJames Feist const std::string* namePtr = 347abf2add6SEd Tanous std::get_if<std::string>(&findName->second); 3485b4aa86bSJames Feist if (namePtr == nullptr) 3495b4aa86bSJames Feist { 3505b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Name Field illegal"; 351b7a08d04SJames Feist messages::internalError(asyncResp->res); 3525b4aa86bSJames Feist return; 3535b4aa86bSJames Feist } 3545b4aa86bSJames Feist std::string name = *namePtr; 3555b4aa86bSJames Feist dbus::utility::escapePathForDbus(name); 35673df0db0SJames Feist 35773df0db0SJames Feist auto findProfiles = intfPair.second.find("Profiles"); 35873df0db0SJames Feist if (findProfiles != intfPair.second.end()) 35973df0db0SJames Feist { 36073df0db0SJames Feist const std::vector<std::string>* profiles = 36173df0db0SJames Feist std::get_if<std::vector<std::string>>( 36273df0db0SJames Feist &findProfiles->second); 36373df0db0SJames Feist if (profiles == nullptr) 36473df0db0SJames Feist { 36573df0db0SJames Feist BMCWEB_LOG_ERROR << "Pid Profiles Field illegal"; 36673df0db0SJames Feist messages::internalError(asyncResp->res); 36773df0db0SJames Feist return; 36873df0db0SJames Feist } 36973df0db0SJames Feist if (std::find(profiles->begin(), profiles->end(), 37073df0db0SJames Feist currentProfile) == profiles->end()) 37173df0db0SJames Feist { 37273df0db0SJames Feist BMCWEB_LOG_INFO 37373df0db0SJames Feist << name << " not supported in current profile"; 37473df0db0SJames Feist continue; 37573df0db0SJames Feist } 37673df0db0SJames Feist } 377b7a08d04SJames Feist nlohmann::json* config = nullptr; 378c33a90ecSJames Feist 379c33a90ecSJames Feist const std::string* classPtr = nullptr; 380c33a90ecSJames Feist auto findClass = intfPair.second.find("Class"); 381c33a90ecSJames Feist if (findClass != intfPair.second.end()) 382c33a90ecSJames Feist { 383c33a90ecSJames Feist classPtr = std::get_if<std::string>(&findClass->second); 384c33a90ecSJames Feist } 385c33a90ecSJames Feist 3865b4aa86bSJames Feist if (intfPair.first == pidZoneConfigurationIface) 3875b4aa86bSJames Feist { 3885b4aa86bSJames Feist std::string chassis; 3895b4aa86bSJames Feist if (!dbus::utility::getNthStringFromPath( 3905b4aa86bSJames Feist pathPair.first.str, 5, chassis)) 3915b4aa86bSJames Feist { 3925b4aa86bSJames Feist chassis = "#IllegalValue"; 3935b4aa86bSJames Feist } 3945b4aa86bSJames Feist nlohmann::json& zone = zones[name]; 3955b4aa86bSJames Feist zone["Chassis"] = { 3965b4aa86bSJames Feist {"@odata.id", "/redfish/v1/Chassis/" + chassis}}; 3975b4aa86bSJames Feist zone["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/" 3985b4aa86bSJames Feist "OpenBmc/Fan/FanZones/" + 3995b4aa86bSJames Feist name; 4005b4aa86bSJames Feist zone["@odata.type"] = "#OemManager.FanZone"; 401b7a08d04SJames Feist config = &zone; 4025b4aa86bSJames Feist } 4035b4aa86bSJames Feist 404b7a08d04SJames Feist else if (intfPair.first == stepwiseConfigurationIface) 4055b4aa86bSJames Feist { 406c33a90ecSJames Feist if (classPtr == nullptr) 407c33a90ecSJames Feist { 408c33a90ecSJames Feist BMCWEB_LOG_ERROR << "Pid Class Field illegal"; 409c33a90ecSJames Feist messages::internalError(asyncResp->res); 410c33a90ecSJames Feist return; 411c33a90ecSJames Feist } 412c33a90ecSJames Feist 413b7a08d04SJames Feist nlohmann::json& controller = stepwise[name]; 414b7a08d04SJames Feist config = &controller; 4155b4aa86bSJames Feist 416b7a08d04SJames Feist controller["@odata.id"] = 417b7a08d04SJames Feist "/redfish/v1/Managers/bmc#/Oem/" 418b7a08d04SJames Feist "OpenBmc/Fan/StepwiseControllers/" + 419271584abSEd Tanous name; 420b7a08d04SJames Feist controller["@odata.type"] = 421b7a08d04SJames Feist "#OemManager.StepwiseController"; 422b7a08d04SJames Feist 423c33a90ecSJames Feist controller["Direction"] = *classPtr; 4245b4aa86bSJames Feist } 4255b4aa86bSJames Feist 4265b4aa86bSJames Feist // pid and fans are off the same configuration 427b7a08d04SJames Feist else if (intfPair.first == pidConfigurationIface) 4285b4aa86bSJames Feist { 429c33a90ecSJames Feist 4305b4aa86bSJames Feist if (classPtr == nullptr) 4315b4aa86bSJames Feist { 4325b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Class Field illegal"; 433a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 4345b4aa86bSJames Feist return; 4355b4aa86bSJames Feist } 4365b4aa86bSJames Feist bool isFan = *classPtr == "fan"; 4375b4aa86bSJames Feist nlohmann::json& element = 4385b4aa86bSJames Feist isFan ? fans[name] : pids[name]; 439b7a08d04SJames Feist config = &element; 4405b4aa86bSJames Feist if (isFan) 4415b4aa86bSJames Feist { 4425b4aa86bSJames Feist element["@odata.id"] = 4435b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/" 4445b4aa86bSJames Feist "OpenBmc/Fan/FanControllers/" + 445271584abSEd Tanous name; 4465b4aa86bSJames Feist element["@odata.type"] = 4475b4aa86bSJames Feist "#OemManager.FanController"; 4485b4aa86bSJames Feist } 4495b4aa86bSJames Feist else 4505b4aa86bSJames Feist { 4515b4aa86bSJames Feist element["@odata.id"] = 4525b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/" 4535b4aa86bSJames Feist "OpenBmc/Fan/PidControllers/" + 454271584abSEd Tanous name; 4555b4aa86bSJames Feist element["@odata.type"] = 4565b4aa86bSJames Feist "#OemManager.PidController"; 4575b4aa86bSJames Feist } 458b7a08d04SJames Feist } 459b7a08d04SJames Feist else 460b7a08d04SJames Feist { 461b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Unexpected configuration"; 462b7a08d04SJames Feist messages::internalError(asyncResp->res); 463b7a08d04SJames Feist return; 464b7a08d04SJames Feist } 465b7a08d04SJames Feist 466b7a08d04SJames Feist // used for making maps out of 2 vectors 467b7a08d04SJames Feist const std::vector<double>* keys = nullptr; 468b7a08d04SJames Feist const std::vector<double>* values = nullptr; 469b7a08d04SJames Feist 470b7a08d04SJames Feist for (const auto& propertyPair : intfPair.second) 471b7a08d04SJames Feist { 472b7a08d04SJames Feist if (propertyPair.first == "Type" || 473b7a08d04SJames Feist propertyPair.first == "Class" || 474b7a08d04SJames Feist propertyPair.first == "Name") 475b7a08d04SJames Feist { 476b7a08d04SJames Feist continue; 477b7a08d04SJames Feist } 478b7a08d04SJames Feist 479b7a08d04SJames Feist // zones 480b7a08d04SJames Feist if (intfPair.first == pidZoneConfigurationIface) 481b7a08d04SJames Feist { 482b7a08d04SJames Feist const double* ptr = 483abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 484b7a08d04SJames Feist if (ptr == nullptr) 485b7a08d04SJames Feist { 486b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 487b7a08d04SJames Feist << propertyPair.first; 488b7a08d04SJames Feist messages::internalError(asyncResp->res); 489b7a08d04SJames Feist return; 490b7a08d04SJames Feist } 491b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 492b7a08d04SJames Feist } 493b7a08d04SJames Feist 494b7a08d04SJames Feist if (intfPair.first == stepwiseConfigurationIface) 495b7a08d04SJames Feist { 496b7a08d04SJames Feist if (propertyPair.first == "Reading" || 497b7a08d04SJames Feist propertyPair.first == "Output") 498b7a08d04SJames Feist { 499b7a08d04SJames Feist const std::vector<double>* ptr = 500abf2add6SEd Tanous std::get_if<std::vector<double>>( 501b7a08d04SJames Feist &propertyPair.second); 502b7a08d04SJames Feist 503b7a08d04SJames Feist if (ptr == nullptr) 504b7a08d04SJames Feist { 505b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 506b7a08d04SJames Feist << propertyPair.first; 507b7a08d04SJames Feist messages::internalError(asyncResp->res); 508b7a08d04SJames Feist return; 509b7a08d04SJames Feist } 510b7a08d04SJames Feist 511b7a08d04SJames Feist if (propertyPair.first == "Reading") 512b7a08d04SJames Feist { 513b7a08d04SJames Feist keys = ptr; 514b7a08d04SJames Feist } 515b7a08d04SJames Feist else 516b7a08d04SJames Feist { 517b7a08d04SJames Feist values = ptr; 518b7a08d04SJames Feist } 519b7a08d04SJames Feist if (keys && values) 520b7a08d04SJames Feist { 521b7a08d04SJames Feist if (keys->size() != values->size()) 522b7a08d04SJames Feist { 523b7a08d04SJames Feist BMCWEB_LOG_ERROR 524b7a08d04SJames Feist << "Reading and Output size don't " 525b7a08d04SJames Feist "match "; 526b7a08d04SJames Feist messages::internalError(asyncResp->res); 527b7a08d04SJames Feist return; 528b7a08d04SJames Feist } 529b7a08d04SJames Feist nlohmann::json& steps = (*config)["Steps"]; 530b7a08d04SJames Feist steps = nlohmann::json::array(); 531b7a08d04SJames Feist for (size_t ii = 0; ii < keys->size(); ii++) 532b7a08d04SJames Feist { 533b7a08d04SJames Feist steps.push_back( 534b7a08d04SJames Feist {{"Target", (*keys)[ii]}, 535b7a08d04SJames Feist {"Output", (*values)[ii]}}); 536b7a08d04SJames Feist } 537b7a08d04SJames Feist } 538b7a08d04SJames Feist } 539b7a08d04SJames Feist if (propertyPair.first == "NegativeHysteresis" || 540b7a08d04SJames Feist propertyPair.first == "PositiveHysteresis") 541b7a08d04SJames Feist { 542b7a08d04SJames Feist const double* ptr = 543abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 544b7a08d04SJames Feist if (ptr == nullptr) 545b7a08d04SJames Feist { 546b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 547b7a08d04SJames Feist << propertyPair.first; 548b7a08d04SJames Feist messages::internalError(asyncResp->res); 549b7a08d04SJames Feist return; 550b7a08d04SJames Feist } 551b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 552b7a08d04SJames Feist } 553b7a08d04SJames Feist } 554b7a08d04SJames Feist 555b7a08d04SJames Feist // pid and fans are off the same configuration 556b7a08d04SJames Feist if (intfPair.first == pidConfigurationIface || 557b7a08d04SJames Feist intfPair.first == stepwiseConfigurationIface) 558b7a08d04SJames Feist { 5595b4aa86bSJames Feist 5605b4aa86bSJames Feist if (propertyPair.first == "Zones") 5615b4aa86bSJames Feist { 5625b4aa86bSJames Feist const std::vector<std::string>* inputs = 563abf2add6SEd Tanous std::get_if<std::vector<std::string>>( 5641b6b96c5SEd Tanous &propertyPair.second); 5655b4aa86bSJames Feist 5665b4aa86bSJames Feist if (inputs == nullptr) 5675b4aa86bSJames Feist { 5685b4aa86bSJames Feist BMCWEB_LOG_ERROR 5695b4aa86bSJames Feist << "Zones Pid Field Illegal"; 570a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 5715b4aa86bSJames Feist return; 5725b4aa86bSJames Feist } 573b7a08d04SJames Feist auto& data = (*config)[propertyPair.first]; 5745b4aa86bSJames Feist data = nlohmann::json::array(); 5755b4aa86bSJames Feist for (std::string itemCopy : *inputs) 5765b4aa86bSJames Feist { 5775b4aa86bSJames Feist dbus::utility::escapePathForDbus(itemCopy); 5785b4aa86bSJames Feist data.push_back( 5795b4aa86bSJames Feist {{"@odata.id", 5805b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/" 5815b4aa86bSJames Feist "OpenBmc/Fan/FanZones/" + 5825b4aa86bSJames Feist itemCopy}}); 5835b4aa86bSJames Feist } 5845b4aa86bSJames Feist } 5855b4aa86bSJames Feist // todo(james): may never happen, but this 5865b4aa86bSJames Feist // assumes configuration data referenced in the 5875b4aa86bSJames Feist // PID config is provided by the same daemon, we 5885b4aa86bSJames Feist // could add another loop to cover all cases, 5895b4aa86bSJames Feist // but I'm okay kicking this can down the road a 5905b4aa86bSJames Feist // bit 5915b4aa86bSJames Feist 5925b4aa86bSJames Feist else if (propertyPair.first == "Inputs" || 5935b4aa86bSJames Feist propertyPair.first == "Outputs") 5945b4aa86bSJames Feist { 595b7a08d04SJames Feist auto& data = (*config)[propertyPair.first]; 5965b4aa86bSJames Feist const std::vector<std::string>* inputs = 597abf2add6SEd Tanous std::get_if<std::vector<std::string>>( 5981b6b96c5SEd Tanous &propertyPair.second); 5995b4aa86bSJames Feist 6005b4aa86bSJames Feist if (inputs == nullptr) 6015b4aa86bSJames Feist { 6025b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 6035b4aa86bSJames Feist << propertyPair.first; 604f12894f8SJason M. Bills messages::internalError(asyncResp->res); 6055b4aa86bSJames Feist return; 6065b4aa86bSJames Feist } 6075b4aa86bSJames Feist data = *inputs; 608b943aaefSJames Feist } 609b943aaefSJames Feist else if (propertyPair.first == "SetPointOffset") 610b943aaefSJames Feist { 611b943aaefSJames Feist const std::string* ptr = 612b943aaefSJames Feist std::get_if<std::string>( 613b943aaefSJames Feist &propertyPair.second); 614b943aaefSJames Feist 615b943aaefSJames Feist if (ptr == nullptr) 616b943aaefSJames Feist { 617b943aaefSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 618b943aaefSJames Feist << propertyPair.first; 619b943aaefSJames Feist messages::internalError(asyncResp->res); 620b943aaefSJames Feist return; 621b943aaefSJames Feist } 622b943aaefSJames Feist // translate from dbus to redfish 623b943aaefSJames Feist if (*ptr == "WarningHigh") 624b943aaefSJames Feist { 625b943aaefSJames Feist (*config)["SetPointOffset"] = 626b943aaefSJames Feist "UpperThresholdNonCritical"; 627b943aaefSJames Feist } 628b943aaefSJames Feist else if (*ptr == "WarningLow") 629b943aaefSJames Feist { 630b943aaefSJames Feist (*config)["SetPointOffset"] = 631b943aaefSJames Feist "LowerThresholdNonCritical"; 632b943aaefSJames Feist } 633b943aaefSJames Feist else if (*ptr == "CriticalHigh") 634b943aaefSJames Feist { 635b943aaefSJames Feist (*config)["SetPointOffset"] = 636b943aaefSJames Feist "UpperThresholdCritical"; 637b943aaefSJames Feist } 638b943aaefSJames Feist else if (*ptr == "CriticalLow") 639b943aaefSJames Feist { 640b943aaefSJames Feist (*config)["SetPointOffset"] = 641b943aaefSJames Feist "LowerThresholdCritical"; 642b943aaefSJames Feist } 643b943aaefSJames Feist else 644b943aaefSJames Feist { 645b943aaefSJames Feist BMCWEB_LOG_ERROR << "Value Illegal " 646b943aaefSJames Feist << *ptr; 647b943aaefSJames Feist messages::internalError(asyncResp->res); 648b943aaefSJames Feist return; 649b943aaefSJames Feist } 650b943aaefSJames Feist } 651b943aaefSJames Feist // doubles 6525b4aa86bSJames Feist else if (propertyPair.first == 6535b4aa86bSJames Feist "FFGainCoefficient" || 6545b4aa86bSJames Feist propertyPair.first == "FFOffCoefficient" || 6555b4aa86bSJames Feist propertyPair.first == "ICoefficient" || 6565b4aa86bSJames Feist propertyPair.first == "ILimitMax" || 6575b4aa86bSJames Feist propertyPair.first == "ILimitMin" || 658aad1a257SJames Feist propertyPair.first == 659aad1a257SJames Feist "PositiveHysteresis" || 660aad1a257SJames Feist propertyPair.first == 661aad1a257SJames Feist "NegativeHysteresis" || 6625b4aa86bSJames Feist propertyPair.first == "OutLimitMax" || 6635b4aa86bSJames Feist propertyPair.first == "OutLimitMin" || 6645b4aa86bSJames Feist propertyPair.first == "PCoefficient" || 6657625cb81SJames Feist propertyPair.first == "SetPoint" || 6665b4aa86bSJames Feist propertyPair.first == "SlewNeg" || 6675b4aa86bSJames Feist propertyPair.first == "SlewPos") 6685b4aa86bSJames Feist { 6695b4aa86bSJames Feist const double* ptr = 670abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 6715b4aa86bSJames Feist if (ptr == nullptr) 6725b4aa86bSJames Feist { 6735b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 6745b4aa86bSJames Feist << propertyPair.first; 675f12894f8SJason M. Bills messages::internalError(asyncResp->res); 6765b4aa86bSJames Feist return; 6775b4aa86bSJames Feist } 678b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 6795b4aa86bSJames Feist } 6805b4aa86bSJames Feist } 6815b4aa86bSJames Feist } 6825b4aa86bSJames Feist } 6835b4aa86bSJames Feist } 6845b4aa86bSJames Feist }, 6855b4aa86bSJames Feist connection, path, objectManagerIface, "GetManagedObjects"); 6865b4aa86bSJames Feist } 687ca537928SJennifer Lee 68883ff9ab6SJames Feist enum class CreatePIDRet 68983ff9ab6SJames Feist { 69083ff9ab6SJames Feist fail, 69183ff9ab6SJames Feist del, 69283ff9ab6SJames Feist patch 69383ff9ab6SJames Feist }; 69483ff9ab6SJames Feist 695*8d1b46d7Szhanghch05 inline bool 696*8d1b46d7Szhanghch05 getZonesFromJsonReq(const std::shared_ptr<bmcweb::AsyncResp>& response, 6975f2caaefSJames Feist std::vector<nlohmann::json>& config, 6985f2caaefSJames Feist std::vector<std::string>& zones) 6995f2caaefSJames Feist { 700b6baeaa4SJames Feist if (config.empty()) 701b6baeaa4SJames Feist { 702b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Empty Zones"; 703b6baeaa4SJames Feist messages::propertyValueFormatError(response->res, 704b6baeaa4SJames Feist nlohmann::json::array(), "Zones"); 705b6baeaa4SJames Feist return false; 706b6baeaa4SJames Feist } 7075f2caaefSJames Feist for (auto& odata : config) 7085f2caaefSJames Feist { 7095f2caaefSJames Feist std::string path; 7105f2caaefSJames Feist if (!redfish::json_util::readJson(odata, response->res, "@odata.id", 7115f2caaefSJames Feist path)) 7125f2caaefSJames Feist { 7135f2caaefSJames Feist return false; 7145f2caaefSJames Feist } 7155f2caaefSJames Feist std::string input; 71661adbda3SJames Feist 71761adbda3SJames Feist // 8 below comes from 71861adbda3SJames Feist // /redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/Left 71961adbda3SJames Feist // 0 1 2 3 4 5 6 7 8 72061adbda3SJames Feist if (!dbus::utility::getNthStringFromPath(path, 8, input)) 7215f2caaefSJames Feist { 7225f2caaefSJames Feist BMCWEB_LOG_ERROR << "Got invalid path " << path; 7235f2caaefSJames Feist BMCWEB_LOG_ERROR << "Illegal Type Zones"; 7245f2caaefSJames Feist messages::propertyValueFormatError(response->res, odata.dump(), 7255f2caaefSJames Feist "Zones"); 7265f2caaefSJames Feist return false; 7275f2caaefSJames Feist } 7285f2caaefSJames Feist boost::replace_all(input, "_", " "); 7295f2caaefSJames Feist zones.emplace_back(std::move(input)); 7305f2caaefSJames Feist } 7315f2caaefSJames Feist return true; 7325f2caaefSJames Feist } 7335f2caaefSJames Feist 73423a21a1cSEd Tanous inline const dbus::utility::ManagedItem* 73573df0db0SJames Feist findChassis(const dbus::utility::ManagedObjectType& managedObj, 736b6baeaa4SJames Feist const std::string& value, std::string& chassis) 737b6baeaa4SJames Feist { 738b6baeaa4SJames Feist BMCWEB_LOG_DEBUG << "Find Chassis: " << value << "\n"; 739b6baeaa4SJames Feist 740b6baeaa4SJames Feist std::string escaped = boost::replace_all_copy(value, " ", "_"); 741b6baeaa4SJames Feist escaped = "/" + escaped; 742b6baeaa4SJames Feist auto it = std::find_if( 743b6baeaa4SJames Feist managedObj.begin(), managedObj.end(), [&escaped](const auto& obj) { 744b6baeaa4SJames Feist if (boost::algorithm::ends_with(obj.first.str, escaped)) 745b6baeaa4SJames Feist { 746b6baeaa4SJames Feist BMCWEB_LOG_DEBUG << "Matched " << obj.first.str << "\n"; 747b6baeaa4SJames Feist return true; 748b6baeaa4SJames Feist } 749b6baeaa4SJames Feist return false; 750b6baeaa4SJames Feist }); 751b6baeaa4SJames Feist 752b6baeaa4SJames Feist if (it == managedObj.end()) 753b6baeaa4SJames Feist { 75473df0db0SJames Feist return nullptr; 755b6baeaa4SJames Feist } 756b6baeaa4SJames Feist // 5 comes from <chassis-name> being the 5th element 757b6baeaa4SJames Feist // /xyz/openbmc_project/inventory/system/chassis/<chassis-name> 75873df0db0SJames Feist if (dbus::utility::getNthStringFromPath(it->first.str, 5, chassis)) 75973df0db0SJames Feist { 76073df0db0SJames Feist return &(*it); 76173df0db0SJames Feist } 76273df0db0SJames Feist 76373df0db0SJames Feist return nullptr; 764b6baeaa4SJames Feist } 765b6baeaa4SJames Feist 76623a21a1cSEd Tanous inline CreatePIDRet createPidInterface( 767*8d1b46d7Szhanghch05 const std::shared_ptr<bmcweb::AsyncResp>& response, const std::string& type, 768b5a76932SEd Tanous const nlohmann::json::iterator& it, const std::string& path, 76983ff9ab6SJames Feist const dbus::utility::ManagedObjectType& managedObj, bool createNewObject, 77083ff9ab6SJames Feist boost::container::flat_map<std::string, dbus::utility::DbusVariantType>& 77183ff9ab6SJames Feist output, 77273df0db0SJames Feist std::string& chassis, const std::string& profile) 77383ff9ab6SJames Feist { 77483ff9ab6SJames Feist 7755f2caaefSJames Feist // common deleter 776b6baeaa4SJames Feist if (it.value() == nullptr) 7775f2caaefSJames Feist { 7785f2caaefSJames Feist std::string iface; 7795f2caaefSJames Feist if (type == "PidControllers" || type == "FanControllers") 7805f2caaefSJames Feist { 7815f2caaefSJames Feist iface = pidConfigurationIface; 7825f2caaefSJames Feist } 7835f2caaefSJames Feist else if (type == "FanZones") 7845f2caaefSJames Feist { 7855f2caaefSJames Feist iface = pidZoneConfigurationIface; 7865f2caaefSJames Feist } 7875f2caaefSJames Feist else if (type == "StepwiseControllers") 7885f2caaefSJames Feist { 7895f2caaefSJames Feist iface = stepwiseConfigurationIface; 7905f2caaefSJames Feist } 7915f2caaefSJames Feist else 7925f2caaefSJames Feist { 793a0744d38SGunnar Mills BMCWEB_LOG_ERROR << "Illegal Type " << type; 7945f2caaefSJames Feist messages::propertyUnknown(response->res, type); 7955f2caaefSJames Feist return CreatePIDRet::fail; 7965f2caaefSJames Feist } 7976ee7f774SJames Feist 7986ee7f774SJames Feist BMCWEB_LOG_DEBUG << "del " << path << " " << iface << "\n"; 7995f2caaefSJames Feist // delete interface 8005f2caaefSJames Feist crow::connections::systemBus->async_method_call( 8015f2caaefSJames Feist [response, path](const boost::system::error_code ec) { 8025f2caaefSJames Feist if (ec) 8035f2caaefSJames Feist { 8045f2caaefSJames Feist BMCWEB_LOG_ERROR << "Error patching " << path << ": " << ec; 8055f2caaefSJames Feist messages::internalError(response->res); 806b6baeaa4SJames Feist return; 8075f2caaefSJames Feist } 808b6baeaa4SJames Feist messages::success(response->res); 8095f2caaefSJames Feist }, 8105f2caaefSJames Feist "xyz.openbmc_project.EntityManager", path, iface, "Delete"); 8115f2caaefSJames Feist return CreatePIDRet::del; 8125f2caaefSJames Feist } 8135f2caaefSJames Feist 81473df0db0SJames Feist const dbus::utility::ManagedItem* managedItem = nullptr; 815b6baeaa4SJames Feist if (!createNewObject) 816b6baeaa4SJames Feist { 817b6baeaa4SJames Feist // if we aren't creating a new object, we should be able to find it on 818b6baeaa4SJames Feist // d-bus 81973df0db0SJames Feist managedItem = findChassis(managedObj, it.key(), chassis); 82073df0db0SJames Feist if (managedItem == nullptr) 821b6baeaa4SJames Feist { 822b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 823b6baeaa4SJames Feist messages::invalidObject(response->res, it.key()); 824b6baeaa4SJames Feist return CreatePIDRet::fail; 825b6baeaa4SJames Feist } 826b6baeaa4SJames Feist } 827b6baeaa4SJames Feist 82873df0db0SJames Feist if (profile.size() && 82973df0db0SJames Feist (type == "PidControllers" || type == "FanControllers" || 83073df0db0SJames Feist type == "StepwiseControllers")) 83173df0db0SJames Feist { 83273df0db0SJames Feist if (managedItem == nullptr) 83373df0db0SJames Feist { 83473df0db0SJames Feist output["Profiles"] = std::vector<std::string>{profile}; 83573df0db0SJames Feist } 83673df0db0SJames Feist else 83773df0db0SJames Feist { 83873df0db0SJames Feist std::string interface; 83973df0db0SJames Feist if (type == "StepwiseControllers") 84073df0db0SJames Feist { 84173df0db0SJames Feist interface = stepwiseConfigurationIface; 84273df0db0SJames Feist } 84373df0db0SJames Feist else 84473df0db0SJames Feist { 84573df0db0SJames Feist interface = pidConfigurationIface; 84673df0db0SJames Feist } 84773df0db0SJames Feist auto findConfig = managedItem->second.find(interface); 84873df0db0SJames Feist if (findConfig == managedItem->second.end()) 84973df0db0SJames Feist { 85073df0db0SJames Feist BMCWEB_LOG_ERROR 85173df0db0SJames Feist << "Failed to find interface in managed object"; 85273df0db0SJames Feist messages::internalError(response->res); 85373df0db0SJames Feist return CreatePIDRet::fail; 85473df0db0SJames Feist } 85573df0db0SJames Feist auto findProfiles = findConfig->second.find("Profiles"); 85673df0db0SJames Feist if (findProfiles != findConfig->second.end()) 85773df0db0SJames Feist { 85873df0db0SJames Feist const std::vector<std::string>* curProfiles = 85973df0db0SJames Feist std::get_if<std::vector<std::string>>( 86073df0db0SJames Feist &(findProfiles->second)); 86173df0db0SJames Feist if (curProfiles == nullptr) 86273df0db0SJames Feist { 86373df0db0SJames Feist BMCWEB_LOG_ERROR << "Illegal profiles in managed object"; 86473df0db0SJames Feist messages::internalError(response->res); 86573df0db0SJames Feist return CreatePIDRet::fail; 86673df0db0SJames Feist } 86773df0db0SJames Feist if (std::find(curProfiles->begin(), curProfiles->end(), 86873df0db0SJames Feist profile) == curProfiles->end()) 86973df0db0SJames Feist { 87073df0db0SJames Feist std::vector<std::string> newProfiles = *curProfiles; 87173df0db0SJames Feist newProfiles.push_back(profile); 87273df0db0SJames Feist output["Profiles"] = newProfiles; 87373df0db0SJames Feist } 87473df0db0SJames Feist } 87573df0db0SJames Feist } 87673df0db0SJames Feist } 87773df0db0SJames Feist 87883ff9ab6SJames Feist if (type == "PidControllers" || type == "FanControllers") 87983ff9ab6SJames Feist { 88083ff9ab6SJames Feist if (createNewObject) 88183ff9ab6SJames Feist { 88283ff9ab6SJames Feist output["Class"] = type == "PidControllers" ? std::string("temp") 88383ff9ab6SJames Feist : std::string("fan"); 88483ff9ab6SJames Feist output["Type"] = std::string("Pid"); 88583ff9ab6SJames Feist } 8865f2caaefSJames Feist 8875f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> zones; 8885f2caaefSJames Feist std::optional<std::vector<std::string>> inputs; 8895f2caaefSJames Feist std::optional<std::vector<std::string>> outputs; 8905f2caaefSJames Feist std::map<std::string, std::optional<double>> doubles; 891b943aaefSJames Feist std::optional<std::string> setpointOffset; 8925f2caaefSJames Feist if (!redfish::json_util::readJson( 893b6baeaa4SJames Feist it.value(), response->res, "Inputs", inputs, "Outputs", outputs, 8945f2caaefSJames Feist "Zones", zones, "FFGainCoefficient", 8955f2caaefSJames Feist doubles["FFGainCoefficient"], "FFOffCoefficient", 8965f2caaefSJames Feist doubles["FFOffCoefficient"], "ICoefficient", 8975f2caaefSJames Feist doubles["ICoefficient"], "ILimitMax", doubles["ILimitMax"], 8985f2caaefSJames Feist "ILimitMin", doubles["ILimitMin"], "OutLimitMax", 8995f2caaefSJames Feist doubles["OutLimitMax"], "OutLimitMin", doubles["OutLimitMin"], 9005f2caaefSJames Feist "PCoefficient", doubles["PCoefficient"], "SetPoint", 901b943aaefSJames Feist doubles["SetPoint"], "SetPointOffset", setpointOffset, 902b943aaefSJames Feist "SlewNeg", doubles["SlewNeg"], "SlewPos", doubles["SlewPos"], 903b943aaefSJames Feist "PositiveHysteresis", doubles["PositiveHysteresis"], 904b943aaefSJames Feist "NegativeHysteresis", doubles["NegativeHysteresis"])) 90583ff9ab6SJames Feist { 90671f52d96SEd Tanous BMCWEB_LOG_ERROR 90771f52d96SEd Tanous << "Illegal Property " 90871f52d96SEd Tanous << it.value().dump(2, ' ', true, 90971f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 9105f2caaefSJames Feist return CreatePIDRet::fail; 91183ff9ab6SJames Feist } 9125f2caaefSJames Feist if (zones) 9135f2caaefSJames Feist { 9145f2caaefSJames Feist std::vector<std::string> zonesStr; 9155f2caaefSJames Feist if (!getZonesFromJsonReq(response, *zones, zonesStr)) 9165f2caaefSJames Feist { 917a0744d38SGunnar Mills BMCWEB_LOG_ERROR << "Illegal Zones"; 9185f2caaefSJames Feist return CreatePIDRet::fail; 9195f2caaefSJames Feist } 920b6baeaa4SJames Feist if (chassis.empty() && 921b6baeaa4SJames Feist !findChassis(managedObj, zonesStr[0], chassis)) 922b6baeaa4SJames Feist { 923b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 924b6baeaa4SJames Feist messages::invalidObject(response->res, it.key()); 925b6baeaa4SJames Feist return CreatePIDRet::fail; 926b6baeaa4SJames Feist } 927b6baeaa4SJames Feist 9285f2caaefSJames Feist output["Zones"] = std::move(zonesStr); 9295f2caaefSJames Feist } 9305f2caaefSJames Feist if (inputs || outputs) 9315f2caaefSJames Feist { 9325f2caaefSJames Feist std::array<std::optional<std::vector<std::string>>*, 2> containers = 9335f2caaefSJames Feist {&inputs, &outputs}; 9345f2caaefSJames Feist size_t index = 0; 9355f2caaefSJames Feist for (const auto& containerPtr : containers) 9365f2caaefSJames Feist { 9375f2caaefSJames Feist std::optional<std::vector<std::string>>& container = 9385f2caaefSJames Feist *containerPtr; 9395f2caaefSJames Feist if (!container) 9405f2caaefSJames Feist { 9415f2caaefSJames Feist index++; 9425f2caaefSJames Feist continue; 94383ff9ab6SJames Feist } 94483ff9ab6SJames Feist 9455f2caaefSJames Feist for (std::string& value : *container) 94683ff9ab6SJames Feist { 9475f2caaefSJames Feist boost::replace_all(value, "_", " "); 94883ff9ab6SJames Feist } 9495f2caaefSJames Feist std::string key; 9505f2caaefSJames Feist if (index == 0) 9515f2caaefSJames Feist { 9525f2caaefSJames Feist key = "Inputs"; 9535f2caaefSJames Feist } 9545f2caaefSJames Feist else 9555f2caaefSJames Feist { 9565f2caaefSJames Feist key = "Outputs"; 9575f2caaefSJames Feist } 9585f2caaefSJames Feist output[key] = *container; 9595f2caaefSJames Feist index++; 9605f2caaefSJames Feist } 96183ff9ab6SJames Feist } 96283ff9ab6SJames Feist 963b943aaefSJames Feist if (setpointOffset) 964b943aaefSJames Feist { 965b943aaefSJames Feist // translate between redfish and dbus names 966b943aaefSJames Feist if (*setpointOffset == "UpperThresholdNonCritical") 967b943aaefSJames Feist { 968b943aaefSJames Feist output["SetPointOffset"] = std::string("WarningLow"); 969b943aaefSJames Feist } 970b943aaefSJames Feist else if (*setpointOffset == "LowerThresholdNonCritical") 971b943aaefSJames Feist { 972b943aaefSJames Feist output["SetPointOffset"] = std::string("WarningHigh"); 973b943aaefSJames Feist } 974b943aaefSJames Feist else if (*setpointOffset == "LowerThresholdCritical") 975b943aaefSJames Feist { 976b943aaefSJames Feist output["SetPointOffset"] = std::string("CriticalLow"); 977b943aaefSJames Feist } 978b943aaefSJames Feist else if (*setpointOffset == "UpperThresholdCritical") 979b943aaefSJames Feist { 980b943aaefSJames Feist output["SetPointOffset"] = std::string("CriticalHigh"); 981b943aaefSJames Feist } 982b943aaefSJames Feist else 983b943aaefSJames Feist { 984b943aaefSJames Feist BMCWEB_LOG_ERROR << "Invalid setpointoffset " 985b943aaefSJames Feist << *setpointOffset; 986b943aaefSJames Feist messages::invalidObject(response->res, it.key()); 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; 10415f2caaefSJames Feist messages::invalidObject(response->res, chassisId); 104283ff9ab6SJames Feist return CreatePIDRet::fail; 104383ff9ab6SJames Feist } 104483ff9ab6SJames Feist } 1045d3ec07f8SJames Feist if (minThermalOutput) 104683ff9ab6SJames Feist { 1047d3ec07f8SJames Feist output["MinThermalOutput"] = *minThermalOutput; 10485f2caaefSJames Feist } 10495f2caaefSJames Feist if (failSafePercent) 105083ff9ab6SJames Feist { 10515f2caaefSJames Feist output["FailSafePercent"] = *failSafePercent; 10525f2caaefSJames Feist } 10535f2caaefSJames Feist } 10545f2caaefSJames Feist else if (type == "StepwiseControllers") 10555f2caaefSJames Feist { 10565f2caaefSJames Feist output["Type"] = std::string("Stepwise"); 10575f2caaefSJames Feist 10585f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> zones; 10595f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> steps; 10605f2caaefSJames Feist std::optional<std::vector<std::string>> inputs; 10615f2caaefSJames Feist std::optional<double> positiveHysteresis; 10625f2caaefSJames Feist std::optional<double> negativeHysteresis; 1063c33a90ecSJames Feist std::optional<std::string> direction; // upper clipping curve vs lower 10645f2caaefSJames Feist if (!redfish::json_util::readJson( 1065b6baeaa4SJames Feist it.value(), response->res, "Zones", zones, "Steps", steps, 1066b6baeaa4SJames Feist "Inputs", inputs, "PositiveHysteresis", positiveHysteresis, 1067c33a90ecSJames Feist "NegativeHysteresis", negativeHysteresis, "Direction", 1068c33a90ecSJames Feist direction)) 10695f2caaefSJames Feist { 107071f52d96SEd Tanous BMCWEB_LOG_ERROR 107171f52d96SEd Tanous << "Illegal Property " 107271f52d96SEd Tanous << it.value().dump(2, ' ', true, 107371f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 107483ff9ab6SJames Feist return CreatePIDRet::fail; 107583ff9ab6SJames Feist } 10765f2caaefSJames Feist 10775f2caaefSJames Feist if (zones) 107883ff9ab6SJames Feist { 1079b6baeaa4SJames Feist std::vector<std::string> zonesStrs; 1080b6baeaa4SJames Feist if (!getZonesFromJsonReq(response, *zones, zonesStrs)) 10815f2caaefSJames Feist { 1082a0744d38SGunnar Mills BMCWEB_LOG_ERROR << "Illegal Zones"; 108383ff9ab6SJames Feist return CreatePIDRet::fail; 108483ff9ab6SJames Feist } 1085b6baeaa4SJames Feist if (chassis.empty() && 1086b6baeaa4SJames Feist !findChassis(managedObj, zonesStrs[0], chassis)) 1087b6baeaa4SJames Feist { 1088b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 1089b6baeaa4SJames Feist messages::invalidObject(response->res, it.key()); 1090b6baeaa4SJames Feist return CreatePIDRet::fail; 1091b6baeaa4SJames Feist } 1092b6baeaa4SJames Feist output["Zones"] = std::move(zonesStrs); 10935f2caaefSJames Feist } 10945f2caaefSJames Feist if (steps) 10955f2caaefSJames Feist { 10965f2caaefSJames Feist std::vector<double> readings; 10975f2caaefSJames Feist std::vector<double> outputs; 10985f2caaefSJames Feist for (auto& step : *steps) 10995f2caaefSJames Feist { 11005f2caaefSJames Feist double target; 110123a21a1cSEd Tanous double out; 11025f2caaefSJames Feist 11035f2caaefSJames Feist if (!redfish::json_util::readJson(step, response->res, "Target", 110423a21a1cSEd Tanous target, "Output", out)) 11055f2caaefSJames Feist { 110671f52d96SEd Tanous BMCWEB_LOG_ERROR 110771f52d96SEd Tanous << "Illegal Property " 110871f52d96SEd Tanous << it.value().dump( 110971f52d96SEd Tanous 2, ' ', true, 111071f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 11115f2caaefSJames Feist return CreatePIDRet::fail; 11125f2caaefSJames Feist } 11135f2caaefSJames Feist readings.emplace_back(target); 111423a21a1cSEd Tanous outputs.emplace_back(out); 11155f2caaefSJames Feist } 11165f2caaefSJames Feist output["Reading"] = std::move(readings); 11175f2caaefSJames Feist output["Output"] = std::move(outputs); 11185f2caaefSJames Feist } 11195f2caaefSJames Feist if (inputs) 11205f2caaefSJames Feist { 11215f2caaefSJames Feist for (std::string& value : *inputs) 11225f2caaefSJames Feist { 11235f2caaefSJames Feist boost::replace_all(value, "_", " "); 11245f2caaefSJames Feist } 11255f2caaefSJames Feist output["Inputs"] = std::move(*inputs); 11265f2caaefSJames Feist } 11275f2caaefSJames Feist if (negativeHysteresis) 11285f2caaefSJames Feist { 11295f2caaefSJames Feist output["NegativeHysteresis"] = *negativeHysteresis; 11305f2caaefSJames Feist } 11315f2caaefSJames Feist if (positiveHysteresis) 11325f2caaefSJames Feist { 11335f2caaefSJames Feist output["PositiveHysteresis"] = *positiveHysteresis; 113483ff9ab6SJames Feist } 1135c33a90ecSJames Feist if (direction) 1136c33a90ecSJames Feist { 1137c33a90ecSJames Feist constexpr const std::array<const char*, 2> allowedDirections = { 1138c33a90ecSJames Feist "Ceiling", "Floor"}; 1139c33a90ecSJames Feist if (std::find(allowedDirections.begin(), allowedDirections.end(), 1140c33a90ecSJames Feist *direction) == allowedDirections.end()) 1141c33a90ecSJames Feist { 1142c33a90ecSJames Feist messages::propertyValueTypeError(response->res, "Direction", 1143c33a90ecSJames Feist *direction); 1144c33a90ecSJames Feist return CreatePIDRet::fail; 1145c33a90ecSJames Feist } 1146c33a90ecSJames Feist output["Class"] = *direction; 1147c33a90ecSJames Feist } 114883ff9ab6SJames Feist } 114983ff9ab6SJames Feist else 115083ff9ab6SJames Feist { 1151a0744d38SGunnar Mills BMCWEB_LOG_ERROR << "Illegal Type " << type; 115235a62c7cSJason M. Bills messages::propertyUnknown(response->res, type); 115383ff9ab6SJames Feist return CreatePIDRet::fail; 115483ff9ab6SJames Feist } 115583ff9ab6SJames Feist return CreatePIDRet::patch; 115683ff9ab6SJames Feist } 115773df0db0SJames Feist struct GetPIDValues : std::enable_shared_from_this<GetPIDValues> 115873df0db0SJames Feist { 115983ff9ab6SJames Feist 1160*8d1b46d7Szhanghch05 GetPIDValues(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn) : 116123a21a1cSEd Tanous asyncResp(asyncRespIn) 116273df0db0SJames Feist 11631214b7e7SGunnar Mills {} 11649c310685SBorawski.Lukasz 116573df0db0SJames Feist void run() 11665b4aa86bSJames Feist { 116773df0db0SJames Feist std::shared_ptr<GetPIDValues> self = shared_from_this(); 116873df0db0SJames Feist 116973df0db0SJames Feist // get all configurations 11705b4aa86bSJames Feist crow::connections::systemBus->async_method_call( 117173df0db0SJames Feist [self](const boost::system::error_code ec, 117223a21a1cSEd Tanous const crow::openbmc_mapper::GetSubTreeType& subtreeLocal) { 11735b4aa86bSJames Feist if (ec) 11745b4aa86bSJames Feist { 11755b4aa86bSJames Feist BMCWEB_LOG_ERROR << ec; 117673df0db0SJames Feist messages::internalError(self->asyncResp->res); 117773df0db0SJames Feist return; 117873df0db0SJames Feist } 117923a21a1cSEd Tanous self->subtree = subtreeLocal; 118073df0db0SJames Feist }, 118173df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", 118273df0db0SJames Feist "/xyz/openbmc_project/object_mapper", 118373df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 118473df0db0SJames Feist std::array<const char*, 4>{ 118573df0db0SJames Feist pidConfigurationIface, pidZoneConfigurationIface, 118673df0db0SJames Feist objectManagerIface, stepwiseConfigurationIface}); 118773df0db0SJames Feist 118873df0db0SJames Feist // at the same time get the selected profile 118973df0db0SJames Feist crow::connections::systemBus->async_method_call( 119073df0db0SJames Feist [self](const boost::system::error_code ec, 119123a21a1cSEd Tanous const crow::openbmc_mapper::GetSubTreeType& subtreeLocal) { 119223a21a1cSEd Tanous if (ec || subtreeLocal.empty()) 119373df0db0SJames Feist { 119473df0db0SJames Feist return; 119573df0db0SJames Feist } 119623a21a1cSEd Tanous if (subtreeLocal[0].second.size() != 1) 119773df0db0SJames Feist { 119873df0db0SJames Feist // invalid mapper response, should never happen 119973df0db0SJames Feist BMCWEB_LOG_ERROR << "GetPIDValues: Mapper Error"; 120073df0db0SJames Feist messages::internalError(self->asyncResp->res); 12015b4aa86bSJames Feist return; 12025b4aa86bSJames Feist } 12035b4aa86bSJames Feist 120423a21a1cSEd Tanous const std::string& path = subtreeLocal[0].first; 120523a21a1cSEd Tanous const std::string& owner = subtreeLocal[0].second[0].first; 120673df0db0SJames Feist crow::connections::systemBus->async_method_call( 120773df0db0SJames Feist [path, owner, self]( 120823a21a1cSEd Tanous const boost::system::error_code ec2, 120973df0db0SJames Feist const boost::container::flat_map< 121073df0db0SJames Feist std::string, std::variant<std::vector<std::string>, 121173df0db0SJames Feist std::string>>& resp) { 121223a21a1cSEd Tanous if (ec2) 121373df0db0SJames Feist { 121473df0db0SJames Feist BMCWEB_LOG_ERROR << "GetPIDValues: Can't get " 121573df0db0SJames Feist "thermalModeIface " 121673df0db0SJames Feist << path; 121773df0db0SJames Feist messages::internalError(self->asyncResp->res); 121873df0db0SJames Feist return; 121973df0db0SJames Feist } 1220271584abSEd Tanous const std::string* current = nullptr; 1221271584abSEd Tanous const std::vector<std::string>* supported = nullptr; 122273df0db0SJames Feist for (auto& [key, value] : resp) 122373df0db0SJames Feist { 122473df0db0SJames Feist if (key == "Current") 122573df0db0SJames Feist { 122673df0db0SJames Feist current = std::get_if<std::string>(&value); 122773df0db0SJames Feist if (current == nullptr) 122873df0db0SJames Feist { 122973df0db0SJames Feist BMCWEB_LOG_ERROR 123073df0db0SJames Feist << "GetPIDValues: thermal mode " 123173df0db0SJames Feist "iface invalid " 123273df0db0SJames Feist << path; 123373df0db0SJames Feist messages::internalError( 123473df0db0SJames Feist self->asyncResp->res); 123573df0db0SJames Feist return; 123673df0db0SJames Feist } 123773df0db0SJames Feist } 123873df0db0SJames Feist if (key == "Supported") 123973df0db0SJames Feist { 124073df0db0SJames Feist supported = 124173df0db0SJames Feist std::get_if<std::vector<std::string>>( 124273df0db0SJames Feist &value); 124373df0db0SJames Feist if (supported == nullptr) 124473df0db0SJames Feist { 124573df0db0SJames Feist BMCWEB_LOG_ERROR 124673df0db0SJames Feist << "GetPIDValues: thermal mode " 124773df0db0SJames Feist "iface invalid" 124873df0db0SJames Feist << path; 124973df0db0SJames Feist messages::internalError( 125073df0db0SJames Feist self->asyncResp->res); 125173df0db0SJames Feist return; 125273df0db0SJames Feist } 125373df0db0SJames Feist } 125473df0db0SJames Feist } 125573df0db0SJames Feist if (current == nullptr || supported == nullptr) 125673df0db0SJames Feist { 125773df0db0SJames Feist BMCWEB_LOG_ERROR << "GetPIDValues: thermal mode " 125873df0db0SJames Feist "iface invalid " 125973df0db0SJames Feist << path; 126073df0db0SJames Feist messages::internalError(self->asyncResp->res); 126173df0db0SJames Feist return; 126273df0db0SJames Feist } 126373df0db0SJames Feist self->currentProfile = *current; 126473df0db0SJames Feist self->supportedProfiles = *supported; 126573df0db0SJames Feist }, 126673df0db0SJames Feist owner, path, "org.freedesktop.DBus.Properties", "GetAll", 126773df0db0SJames Feist thermalModeIface); 126873df0db0SJames Feist }, 126973df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", 127073df0db0SJames Feist "/xyz/openbmc_project/object_mapper", 127173df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 127273df0db0SJames Feist std::array<const char*, 1>{thermalModeIface}); 127373df0db0SJames Feist } 127473df0db0SJames Feist 127573df0db0SJames Feist ~GetPIDValues() 127673df0db0SJames Feist { 127773df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 127873df0db0SJames Feist { 127973df0db0SJames Feist return; 128073df0db0SJames Feist } 12815b4aa86bSJames Feist // create map of <connection, path to objMgr>> 128273df0db0SJames Feist boost::container::flat_map<std::string, std::string> objectMgrPaths; 12836bce33bcSJames Feist boost::container::flat_set<std::string> calledConnections; 12845b4aa86bSJames Feist for (const auto& pathGroup : subtree) 12855b4aa86bSJames Feist { 12865b4aa86bSJames Feist for (const auto& connectionGroup : pathGroup.second) 12875b4aa86bSJames Feist { 12886bce33bcSJames Feist auto findConnection = 12896bce33bcSJames Feist calledConnections.find(connectionGroup.first); 12906bce33bcSJames Feist if (findConnection != calledConnections.end()) 12916bce33bcSJames Feist { 12926bce33bcSJames Feist break; 12936bce33bcSJames Feist } 129473df0db0SJames Feist for (const std::string& interface : connectionGroup.second) 12955b4aa86bSJames Feist { 12965b4aa86bSJames Feist if (interface == objectManagerIface) 12975b4aa86bSJames Feist { 129873df0db0SJames Feist objectMgrPaths[connectionGroup.first] = pathGroup.first; 12995b4aa86bSJames Feist } 13005b4aa86bSJames Feist // this list is alphabetical, so we 13015b4aa86bSJames Feist // should have found the objMgr by now 13025b4aa86bSJames Feist if (interface == pidConfigurationIface || 1303b7a08d04SJames Feist interface == pidZoneConfigurationIface || 1304b7a08d04SJames Feist interface == stepwiseConfigurationIface) 13055b4aa86bSJames Feist { 13065b4aa86bSJames Feist auto findObjMgr = 13075b4aa86bSJames Feist objectMgrPaths.find(connectionGroup.first); 13085b4aa86bSJames Feist if (findObjMgr == objectMgrPaths.end()) 13095b4aa86bSJames Feist { 13105b4aa86bSJames Feist BMCWEB_LOG_DEBUG << connectionGroup.first 13115b4aa86bSJames Feist << "Has no Object Manager"; 13125b4aa86bSJames Feist continue; 13135b4aa86bSJames Feist } 13146bce33bcSJames Feist 13156bce33bcSJames Feist calledConnections.insert(connectionGroup.first); 13166bce33bcSJames Feist 131773df0db0SJames Feist asyncPopulatePid(findObjMgr->first, findObjMgr->second, 131873df0db0SJames Feist currentProfile, supportedProfiles, 131973df0db0SJames Feist asyncResp); 13205b4aa86bSJames Feist break; 13215b4aa86bSJames Feist } 13225b4aa86bSJames Feist } 13235b4aa86bSJames Feist } 13245b4aa86bSJames Feist } 132573df0db0SJames Feist } 132673df0db0SJames Feist 132773df0db0SJames Feist std::vector<std::string> supportedProfiles; 132873df0db0SJames Feist std::string currentProfile; 132973df0db0SJames Feist crow::openbmc_mapper::GetSubTreeType subtree; 1330*8d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> asyncResp; 133173df0db0SJames Feist }; 133273df0db0SJames Feist 133373df0db0SJames Feist struct SetPIDValues : std::enable_shared_from_this<SetPIDValues> 133473df0db0SJames Feist { 133573df0db0SJames Feist 1336*8d1b46d7Szhanghch05 SetPIDValues(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn, 133773df0db0SJames Feist nlohmann::json& data) : 1338271584abSEd Tanous asyncResp(asyncRespIn) 133973df0db0SJames Feist { 134073df0db0SJames Feist 134173df0db0SJames Feist std::optional<nlohmann::json> pidControllers; 134273df0db0SJames Feist std::optional<nlohmann::json> fanControllers; 134373df0db0SJames Feist std::optional<nlohmann::json> fanZones; 134473df0db0SJames Feist std::optional<nlohmann::json> stepwiseControllers; 134573df0db0SJames Feist 134673df0db0SJames Feist if (!redfish::json_util::readJson( 134773df0db0SJames Feist data, asyncResp->res, "PidControllers", pidControllers, 134873df0db0SJames Feist "FanControllers", fanControllers, "FanZones", fanZones, 134973df0db0SJames Feist "StepwiseControllers", stepwiseControllers, "Profile", profile)) 135073df0db0SJames Feist { 135171f52d96SEd Tanous BMCWEB_LOG_ERROR 135271f52d96SEd Tanous << "Illegal Property " 135371f52d96SEd Tanous << data.dump(2, ' ', true, 135471f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 135573df0db0SJames Feist return; 135673df0db0SJames Feist } 135773df0db0SJames Feist configuration.emplace_back("PidControllers", std::move(pidControllers)); 135873df0db0SJames Feist configuration.emplace_back("FanControllers", std::move(fanControllers)); 135973df0db0SJames Feist configuration.emplace_back("FanZones", std::move(fanZones)); 136073df0db0SJames Feist configuration.emplace_back("StepwiseControllers", 136173df0db0SJames Feist std::move(stepwiseControllers)); 136273df0db0SJames Feist } 136373df0db0SJames Feist void run() 136473df0db0SJames Feist { 136573df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 136673df0db0SJames Feist { 136773df0db0SJames Feist return; 136873df0db0SJames Feist } 136973df0db0SJames Feist 137073df0db0SJames Feist std::shared_ptr<SetPIDValues> self = shared_from_this(); 137173df0db0SJames Feist 137273df0db0SJames Feist // todo(james): might make sense to do a mapper call here if this 137373df0db0SJames Feist // interface gets more traction 137473df0db0SJames Feist crow::connections::systemBus->async_method_call( 137573df0db0SJames Feist [self](const boost::system::error_code ec, 1376271584abSEd Tanous dbus::utility::ManagedObjectType& mObj) { 137773df0db0SJames Feist if (ec) 137873df0db0SJames Feist { 137973df0db0SJames Feist BMCWEB_LOG_ERROR << "Error communicating to Entity Manager"; 138073df0db0SJames Feist messages::internalError(self->asyncResp->res); 138173df0db0SJames Feist return; 138273df0db0SJames Feist } 1383e69d9de2SJames Feist const std::array<const char*, 3> configurations = { 1384e69d9de2SJames Feist pidConfigurationIface, pidZoneConfigurationIface, 1385e69d9de2SJames Feist stepwiseConfigurationIface}; 1386e69d9de2SJames Feist 138714b0b8d5SJames Feist for (const auto& [path, object] : mObj) 1388e69d9de2SJames Feist { 138914b0b8d5SJames Feist for (const auto& [interface, _] : object) 1390e69d9de2SJames Feist { 1391e69d9de2SJames Feist if (std::find(configurations.begin(), 1392e69d9de2SJames Feist configurations.end(), 1393e69d9de2SJames Feist interface) != configurations.end()) 1394e69d9de2SJames Feist { 139514b0b8d5SJames Feist self->objectCount++; 1396e69d9de2SJames Feist break; 1397e69d9de2SJames Feist } 1398e69d9de2SJames Feist } 1399e69d9de2SJames Feist } 1400271584abSEd Tanous self->managedObj = std::move(mObj); 140173df0db0SJames Feist }, 140273df0db0SJames Feist "xyz.openbmc_project.EntityManager", "/", objectManagerIface, 140373df0db0SJames Feist "GetManagedObjects"); 140473df0db0SJames Feist 140573df0db0SJames Feist // at the same time get the profile information 140673df0db0SJames Feist crow::connections::systemBus->async_method_call( 140773df0db0SJames Feist [self](const boost::system::error_code ec, 140873df0db0SJames Feist const crow::openbmc_mapper::GetSubTreeType& subtree) { 140973df0db0SJames Feist if (ec || subtree.empty()) 141073df0db0SJames Feist { 141173df0db0SJames Feist return; 141273df0db0SJames Feist } 141373df0db0SJames Feist if (subtree[0].second.empty()) 141473df0db0SJames Feist { 141573df0db0SJames Feist // invalid mapper response, should never happen 141673df0db0SJames Feist BMCWEB_LOG_ERROR << "SetPIDValues: Mapper Error"; 141773df0db0SJames Feist messages::internalError(self->asyncResp->res); 141873df0db0SJames Feist return; 141973df0db0SJames Feist } 142073df0db0SJames Feist 142173df0db0SJames Feist const std::string& path = subtree[0].first; 142273df0db0SJames Feist const std::string& owner = subtree[0].second[0].first; 142373df0db0SJames Feist crow::connections::systemBus->async_method_call( 142473df0db0SJames Feist [self, path, owner]( 1425cb13a392SEd Tanous const boost::system::error_code ec2, 142673df0db0SJames Feist const boost::container::flat_map< 142773df0db0SJames Feist std::string, std::variant<std::vector<std::string>, 1428271584abSEd Tanous std::string>>& r) { 1429cb13a392SEd Tanous if (ec2) 143073df0db0SJames Feist { 143173df0db0SJames Feist BMCWEB_LOG_ERROR << "SetPIDValues: Can't get " 143273df0db0SJames Feist "thermalModeIface " 143373df0db0SJames Feist << path; 143473df0db0SJames Feist messages::internalError(self->asyncResp->res); 143573df0db0SJames Feist return; 143673df0db0SJames Feist } 1437271584abSEd Tanous const std::string* current = nullptr; 1438271584abSEd Tanous const std::vector<std::string>* supported = nullptr; 1439271584abSEd Tanous for (auto& [key, value] : r) 144073df0db0SJames Feist { 144173df0db0SJames Feist if (key == "Current") 144273df0db0SJames Feist { 144373df0db0SJames Feist current = std::get_if<std::string>(&value); 144473df0db0SJames Feist if (current == nullptr) 144573df0db0SJames Feist { 144673df0db0SJames Feist BMCWEB_LOG_ERROR 144773df0db0SJames Feist << "SetPIDValues: thermal mode " 144873df0db0SJames Feist "iface invalid " 144973df0db0SJames Feist << path; 145073df0db0SJames Feist messages::internalError( 145173df0db0SJames Feist self->asyncResp->res); 145273df0db0SJames Feist return; 145373df0db0SJames Feist } 145473df0db0SJames Feist } 145573df0db0SJames Feist if (key == "Supported") 145673df0db0SJames Feist { 145773df0db0SJames Feist supported = 145873df0db0SJames Feist std::get_if<std::vector<std::string>>( 145973df0db0SJames Feist &value); 146073df0db0SJames Feist if (supported == nullptr) 146173df0db0SJames Feist { 146273df0db0SJames Feist BMCWEB_LOG_ERROR 146373df0db0SJames Feist << "SetPIDValues: thermal mode " 146473df0db0SJames Feist "iface invalid" 146573df0db0SJames Feist << path; 146673df0db0SJames Feist messages::internalError( 146773df0db0SJames Feist self->asyncResp->res); 146873df0db0SJames Feist return; 146973df0db0SJames Feist } 147073df0db0SJames Feist } 147173df0db0SJames Feist } 147273df0db0SJames Feist if (current == nullptr || supported == nullptr) 147373df0db0SJames Feist { 147473df0db0SJames Feist BMCWEB_LOG_ERROR << "SetPIDValues: thermal mode " 147573df0db0SJames Feist "iface invalid " 147673df0db0SJames Feist << path; 147773df0db0SJames Feist messages::internalError(self->asyncResp->res); 147873df0db0SJames Feist return; 147973df0db0SJames Feist } 148073df0db0SJames Feist self->currentProfile = *current; 148173df0db0SJames Feist self->supportedProfiles = *supported; 148273df0db0SJames Feist self->profileConnection = owner; 148373df0db0SJames Feist self->profilePath = path; 148473df0db0SJames Feist }, 148573df0db0SJames Feist owner, path, "org.freedesktop.DBus.Properties", "GetAll", 148673df0db0SJames Feist thermalModeIface); 14875b4aa86bSJames Feist }, 14885b4aa86bSJames Feist "xyz.openbmc_project.ObjectMapper", 14895b4aa86bSJames Feist "/xyz/openbmc_project/object_mapper", 14905b4aa86bSJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 149173df0db0SJames Feist std::array<const char*, 1>{thermalModeIface}); 149273df0db0SJames Feist } 149373df0db0SJames Feist ~SetPIDValues() 149473df0db0SJames Feist { 149573df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 149673df0db0SJames Feist { 149773df0db0SJames Feist return; 14985b4aa86bSJames Feist } 1499*8d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> response = asyncResp; 150073df0db0SJames Feist if (profile) 150173df0db0SJames Feist { 150273df0db0SJames Feist if (std::find(supportedProfiles.begin(), supportedProfiles.end(), 150373df0db0SJames Feist *profile) == supportedProfiles.end()) 150473df0db0SJames Feist { 150573df0db0SJames Feist messages::actionParameterUnknown(response->res, "Profile", 150673df0db0SJames Feist *profile); 150773df0db0SJames Feist return; 150873df0db0SJames Feist } 150973df0db0SJames Feist currentProfile = *profile; 151073df0db0SJames Feist crow::connections::systemBus->async_method_call( 151173df0db0SJames Feist [response](const boost::system::error_code ec) { 151273df0db0SJames Feist if (ec) 151373df0db0SJames Feist { 151473df0db0SJames Feist BMCWEB_LOG_ERROR << "Error patching profile" << ec; 151573df0db0SJames Feist messages::internalError(response->res); 151673df0db0SJames Feist } 151773df0db0SJames Feist }, 151873df0db0SJames Feist profileConnection, profilePath, 151973df0db0SJames Feist "org.freedesktop.DBus.Properties", "Set", thermalModeIface, 152073df0db0SJames Feist "Current", std::variant<std::string>(*profile)); 152173df0db0SJames Feist } 152273df0db0SJames Feist 152373df0db0SJames Feist for (auto& containerPair : configuration) 152473df0db0SJames Feist { 152573df0db0SJames Feist auto& container = containerPair.second; 152673df0db0SJames Feist if (!container) 152773df0db0SJames Feist { 152873df0db0SJames Feist continue; 152973df0db0SJames Feist } 15306ee7f774SJames Feist BMCWEB_LOG_DEBUG << *container; 15316ee7f774SJames Feist 153273df0db0SJames Feist std::string& type = containerPair.first; 153373df0db0SJames Feist 153473df0db0SJames Feist for (nlohmann::json::iterator it = container->begin(); 153517a897dfSManojkiran Eda it != container->end(); ++it) 153673df0db0SJames Feist { 153773df0db0SJames Feist const auto& name = it.key(); 15386ee7f774SJames Feist BMCWEB_LOG_DEBUG << "looking for " << name; 15396ee7f774SJames Feist 154073df0db0SJames Feist auto pathItr = 154173df0db0SJames Feist std::find_if(managedObj.begin(), managedObj.end(), 154273df0db0SJames Feist [&name](const auto& obj) { 154373df0db0SJames Feist return boost::algorithm::ends_with( 154473df0db0SJames Feist obj.first.str, "/" + name); 154573df0db0SJames Feist }); 154673df0db0SJames Feist boost::container::flat_map<std::string, 154773df0db0SJames Feist dbus::utility::DbusVariantType> 154873df0db0SJames Feist output; 154973df0db0SJames Feist 155073df0db0SJames Feist output.reserve(16); // The pid interface length 155173df0db0SJames Feist 155273df0db0SJames Feist // determines if we're patching entity-manager or 155373df0db0SJames Feist // creating a new object 155473df0db0SJames Feist bool createNewObject = (pathItr == managedObj.end()); 15556ee7f774SJames Feist BMCWEB_LOG_DEBUG << "Found = " << !createNewObject; 15566ee7f774SJames Feist 155773df0db0SJames Feist std::string iface; 155873df0db0SJames Feist if (type == "PidControllers" || type == "FanControllers") 155973df0db0SJames Feist { 156073df0db0SJames Feist iface = pidConfigurationIface; 156173df0db0SJames Feist if (!createNewObject && 156273df0db0SJames Feist pathItr->second.find(pidConfigurationIface) == 156373df0db0SJames Feist pathItr->second.end()) 156473df0db0SJames Feist { 156573df0db0SJames Feist createNewObject = true; 156673df0db0SJames Feist } 156773df0db0SJames Feist } 156873df0db0SJames Feist else if (type == "FanZones") 156973df0db0SJames Feist { 157073df0db0SJames Feist iface = pidZoneConfigurationIface; 157173df0db0SJames Feist if (!createNewObject && 157273df0db0SJames Feist pathItr->second.find(pidZoneConfigurationIface) == 157373df0db0SJames Feist pathItr->second.end()) 157473df0db0SJames Feist { 157573df0db0SJames Feist 157673df0db0SJames Feist createNewObject = true; 157773df0db0SJames Feist } 157873df0db0SJames Feist } 157973df0db0SJames Feist else if (type == "StepwiseControllers") 158073df0db0SJames Feist { 158173df0db0SJames Feist iface = stepwiseConfigurationIface; 158273df0db0SJames Feist if (!createNewObject && 158373df0db0SJames Feist pathItr->second.find(stepwiseConfigurationIface) == 158473df0db0SJames Feist pathItr->second.end()) 158573df0db0SJames Feist { 158673df0db0SJames Feist createNewObject = true; 158773df0db0SJames Feist } 158873df0db0SJames Feist } 15896ee7f774SJames Feist 15906ee7f774SJames Feist if (createNewObject && it.value() == nullptr) 15916ee7f774SJames Feist { 15924e0453b1SGunnar Mills // can't delete a non-existent object 15936ee7f774SJames Feist messages::invalidObject(response->res, name); 15946ee7f774SJames Feist continue; 15956ee7f774SJames Feist } 15966ee7f774SJames Feist 15976ee7f774SJames Feist std::string path; 15986ee7f774SJames Feist if (pathItr != managedObj.end()) 15996ee7f774SJames Feist { 16006ee7f774SJames Feist path = pathItr->first.str; 16016ee7f774SJames Feist } 16026ee7f774SJames Feist 160373df0db0SJames Feist BMCWEB_LOG_DEBUG << "Create new = " << createNewObject << "\n"; 1604e69d9de2SJames Feist 1605e69d9de2SJames Feist // arbitrary limit to avoid attacks 1606e69d9de2SJames Feist constexpr const size_t controllerLimit = 500; 160714b0b8d5SJames Feist if (createNewObject && objectCount >= controllerLimit) 1608e69d9de2SJames Feist { 1609e69d9de2SJames Feist messages::resourceExhaustion(response->res, type); 1610e69d9de2SJames Feist continue; 1611e69d9de2SJames Feist } 1612e69d9de2SJames Feist 161373df0db0SJames Feist output["Name"] = boost::replace_all_copy(name, "_", " "); 161473df0db0SJames Feist 161573df0db0SJames Feist std::string chassis; 161673df0db0SJames Feist CreatePIDRet ret = createPidInterface( 16176ee7f774SJames Feist response, type, it, path, managedObj, createNewObject, 16186ee7f774SJames Feist output, chassis, currentProfile); 161973df0db0SJames Feist if (ret == CreatePIDRet::fail) 162073df0db0SJames Feist { 162173df0db0SJames Feist return; 162273df0db0SJames Feist } 16233174e4dfSEd Tanous if (ret == CreatePIDRet::del) 162473df0db0SJames Feist { 162573df0db0SJames Feist continue; 162673df0db0SJames Feist } 162773df0db0SJames Feist 162873df0db0SJames Feist if (!createNewObject) 162973df0db0SJames Feist { 163073df0db0SJames Feist for (const auto& property : output) 163173df0db0SJames Feist { 163273df0db0SJames Feist crow::connections::systemBus->async_method_call( 163373df0db0SJames Feist [response, 163473df0db0SJames Feist propertyName{std::string(property.first)}]( 163573df0db0SJames Feist const boost::system::error_code ec) { 163673df0db0SJames Feist if (ec) 163773df0db0SJames Feist { 163873df0db0SJames Feist BMCWEB_LOG_ERROR << "Error patching " 163973df0db0SJames Feist << propertyName << ": " 164073df0db0SJames Feist << ec; 164173df0db0SJames Feist messages::internalError(response->res); 164273df0db0SJames Feist return; 164373df0db0SJames Feist } 164473df0db0SJames Feist messages::success(response->res); 164573df0db0SJames Feist }, 16466ee7f774SJames Feist "xyz.openbmc_project.EntityManager", path, 164773df0db0SJames Feist "org.freedesktop.DBus.Properties", "Set", iface, 164873df0db0SJames Feist property.first, property.second); 164973df0db0SJames Feist } 165073df0db0SJames Feist } 165173df0db0SJames Feist else 165273df0db0SJames Feist { 165373df0db0SJames Feist if (chassis.empty()) 165473df0db0SJames Feist { 165573df0db0SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config"; 165673df0db0SJames Feist messages::invalidObject(response->res, name); 165773df0db0SJames Feist return; 165873df0db0SJames Feist } 165973df0db0SJames Feist 166073df0db0SJames Feist bool foundChassis = false; 166173df0db0SJames Feist for (const auto& obj : managedObj) 166273df0db0SJames Feist { 166373df0db0SJames Feist if (boost::algorithm::ends_with(obj.first.str, chassis)) 166473df0db0SJames Feist { 166573df0db0SJames Feist chassis = obj.first.str; 166673df0db0SJames Feist foundChassis = true; 166773df0db0SJames Feist break; 166873df0db0SJames Feist } 166973df0db0SJames Feist } 167073df0db0SJames Feist if (!foundChassis) 167173df0db0SJames Feist { 167273df0db0SJames Feist BMCWEB_LOG_ERROR << "Failed to find chassis on dbus"; 167373df0db0SJames Feist messages::resourceMissingAtURI( 167473df0db0SJames Feist response->res, "/redfish/v1/Chassis/" + chassis); 167573df0db0SJames Feist return; 167673df0db0SJames Feist } 167773df0db0SJames Feist 167873df0db0SJames Feist crow::connections::systemBus->async_method_call( 167973df0db0SJames Feist [response](const boost::system::error_code ec) { 168073df0db0SJames Feist if (ec) 168173df0db0SJames Feist { 168273df0db0SJames Feist BMCWEB_LOG_ERROR << "Error Adding Pid Object " 168373df0db0SJames Feist << ec; 168473df0db0SJames Feist messages::internalError(response->res); 168573df0db0SJames Feist return; 168673df0db0SJames Feist } 168773df0db0SJames Feist messages::success(response->res); 168873df0db0SJames Feist }, 168973df0db0SJames Feist "xyz.openbmc_project.EntityManager", chassis, 169073df0db0SJames Feist "xyz.openbmc_project.AddObject", "AddObject", output); 169173df0db0SJames Feist } 169273df0db0SJames Feist } 169373df0db0SJames Feist } 169473df0db0SJames Feist } 1695*8d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> asyncResp; 169673df0db0SJames Feist std::vector<std::pair<std::string, std::optional<nlohmann::json>>> 169773df0db0SJames Feist configuration; 169873df0db0SJames Feist std::optional<std::string> profile; 169973df0db0SJames Feist dbus::utility::ManagedObjectType managedObj; 170073df0db0SJames Feist std::vector<std::string> supportedProfiles; 170173df0db0SJames Feist std::string currentProfile; 170273df0db0SJames Feist std::string profileConnection; 170373df0db0SJames Feist std::string profilePath; 170414b0b8d5SJames Feist size_t objectCount = 0; 170573df0db0SJames Feist }; 170673df0db0SJames Feist 1707071d8fdfSSunnySrivastava1984 /** 1708071d8fdfSSunnySrivastava1984 * @brief Retrieves BMC manager location data over DBus 1709071d8fdfSSunnySrivastava1984 * 1710071d8fdfSSunnySrivastava1984 * @param[in] aResp Shared pointer for completing asynchronous calls 1711071d8fdfSSunnySrivastava1984 * @param[in] connectionName - service name 1712071d8fdfSSunnySrivastava1984 * @param[in] path - object path 1713071d8fdfSSunnySrivastava1984 * @return none 1714071d8fdfSSunnySrivastava1984 */ 1715*8d1b46d7Szhanghch05 inline void getLocation(const std::shared_ptr<bmcweb::AsyncResp>& aResp, 1716071d8fdfSSunnySrivastava1984 const std::string& connectionName, 1717071d8fdfSSunnySrivastava1984 const std::string& path) 1718071d8fdfSSunnySrivastava1984 { 1719071d8fdfSSunnySrivastava1984 BMCWEB_LOG_DEBUG << "Get BMC manager Location data."; 1720071d8fdfSSunnySrivastava1984 1721071d8fdfSSunnySrivastava1984 crow::connections::systemBus->async_method_call( 1722071d8fdfSSunnySrivastava1984 [aResp](const boost::system::error_code ec, 1723071d8fdfSSunnySrivastava1984 const std::variant<std::string>& property) { 1724071d8fdfSSunnySrivastava1984 if (ec) 1725071d8fdfSSunnySrivastava1984 { 1726071d8fdfSSunnySrivastava1984 BMCWEB_LOG_DEBUG << "DBUS response error for " 1727071d8fdfSSunnySrivastava1984 "Location"; 1728071d8fdfSSunnySrivastava1984 messages::internalError(aResp->res); 1729071d8fdfSSunnySrivastava1984 return; 1730071d8fdfSSunnySrivastava1984 } 1731071d8fdfSSunnySrivastava1984 1732071d8fdfSSunnySrivastava1984 const std::string* value = std::get_if<std::string>(&property); 1733071d8fdfSSunnySrivastava1984 1734071d8fdfSSunnySrivastava1984 if (value == nullptr) 1735071d8fdfSSunnySrivastava1984 { 1736071d8fdfSSunnySrivastava1984 // illegal value 1737071d8fdfSSunnySrivastava1984 messages::internalError(aResp->res); 1738071d8fdfSSunnySrivastava1984 return; 1739071d8fdfSSunnySrivastava1984 } 1740071d8fdfSSunnySrivastava1984 1741071d8fdfSSunnySrivastava1984 aResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] = 1742071d8fdfSSunnySrivastava1984 *value; 1743071d8fdfSSunnySrivastava1984 }, 1744071d8fdfSSunnySrivastava1984 connectionName, path, "org.freedesktop.DBus.Properties", "Get", 1745071d8fdfSSunnySrivastava1984 "xyz.openbmc_project.Inventory.Decorator." 1746071d8fdfSSunnySrivastava1984 "LocationCode", 1747071d8fdfSSunnySrivastava1984 "LocationCode"); 1748071d8fdfSSunnySrivastava1984 } 1749071d8fdfSSunnySrivastava1984 175073df0db0SJames Feist class Manager : public Node 175173df0db0SJames Feist { 175273df0db0SJames Feist public: 175352cc112dSEd Tanous Manager(App& app) : Node(app, "/redfish/v1/Managers/bmc/") 175473df0db0SJames Feist { 175552cc112dSEd Tanous 175652cc112dSEd Tanous uuid = persistent_data::getConfig().systemUuid; 175773df0db0SJames Feist entityPrivileges = { 175873df0db0SJames Feist {boost::beast::http::verb::get, {{"Login"}}}, 175973df0db0SJames Feist {boost::beast::http::verb::head, {{"Login"}}}, 176073df0db0SJames Feist {boost::beast::http::verb::patch, {{"ConfigureManager"}}}, 176173df0db0SJames Feist {boost::beast::http::verb::put, {{"ConfigureManager"}}}, 176273df0db0SJames Feist {boost::beast::http::verb::delete_, {{"ConfigureManager"}}}, 176373df0db0SJames Feist {boost::beast::http::verb::post, {{"ConfigureManager"}}}}; 176473df0db0SJames Feist } 176573df0db0SJames Feist 176673df0db0SJames Feist private: 1767*8d1b46d7Szhanghch05 void doGet(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1768*8d1b46d7Szhanghch05 const crow::Request&, const std::vector<std::string>&) override 17691abe55efSEd Tanous { 1770*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc"; 1771*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["@odata.type"] = "#Manager.v1_11_0.Manager"; 1772*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Id"] = "bmc"; 1773*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Name"] = "OpenBmc Manager"; 1774*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Description"] = 1775*8d1b46d7Szhanghch05 "Baseboard Management Controller"; 1776*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["PowerState"] = "On"; 1777*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Status"] = {{"State", "Enabled"}, 1778*8d1b46d7Szhanghch05 {"Health", "OK"}}; 1779*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["ManagerType"] = "BMC"; 1780*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["UUID"] = systemd_utils::getUuid(); 1781*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["ServiceEntryPointUUID"] = uuid; 1782*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model 17830f74e643SEd Tanous 1784*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["LogServices"] = { 17850f74e643SEd Tanous {"@odata.id", "/redfish/v1/Managers/bmc/LogServices"}}; 17860f74e643SEd Tanous 1787*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["NetworkProtocol"] = { 17880f74e643SEd Tanous {"@odata.id", "/redfish/v1/Managers/bmc/NetworkProtocol"}}; 17890f74e643SEd Tanous 1790*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["EthernetInterfaces"] = { 17910f74e643SEd Tanous {"@odata.id", "/redfish/v1/Managers/bmc/EthernetInterfaces"}}; 1792107077deSPrzemyslaw Czarnowski 1793107077deSPrzemyslaw Czarnowski #ifdef BMCWEB_ENABLE_VM_NBDPROXY 1794*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["VirtualMedia"] = { 1795107077deSPrzemyslaw Czarnowski {"@odata.id", "/redfish/v1/Managers/bmc/VirtualMedia"}}; 1796107077deSPrzemyslaw Czarnowski #endif // BMCWEB_ENABLE_VM_NBDPROXY 1797107077deSPrzemyslaw Czarnowski 17980f74e643SEd Tanous // default oem data 1799*8d1b46d7Szhanghch05 nlohmann::json& oem = asyncResp->res.jsonValue["Oem"]; 18000f74e643SEd Tanous nlohmann::json& oemOpenbmc = oem["OpenBmc"]; 18010f74e643SEd Tanous oem["@odata.type"] = "#OemManager.Oem"; 18020f74e643SEd Tanous oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem"; 18030f74e643SEd Tanous oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc"; 18040f74e643SEd Tanous oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc"; 1805cfcd5f6bSMarri Devender Rao oemOpenbmc["Certificates"] = { 1806cfcd5f6bSMarri Devender Rao {"@odata.id", "/redfish/v1/Managers/bmc/Truststore/Certificates"}}; 18070f74e643SEd Tanous 18082a5c4407SGunnar Mills // Manager.Reset (an action) can be many values, OpenBMC only supports 18092a5c4407SGunnar Mills // BMC reboot. 18102a5c4407SGunnar Mills nlohmann::json& managerReset = 1811*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Actions"]["#Manager.Reset"]; 18122a5c4407SGunnar Mills managerReset["target"] = 1813ed5befbdSJennifer Lee "/redfish/v1/Managers/bmc/Actions/Manager.Reset"; 18141cb1a9e6SAppaRao Puli managerReset["@Redfish.ActionInfo"] = 18151cb1a9e6SAppaRao Puli "/redfish/v1/Managers/bmc/ResetActionInfo"; 1816ca537928SJennifer Lee 18173e40fc74SGunnar Mills // ResetToDefaults (Factory Reset) has values like 18183e40fc74SGunnar Mills // PreserveNetworkAndUsers and PreserveNetwork that aren't supported 18193e40fc74SGunnar Mills // on OpenBMC 18203e40fc74SGunnar Mills nlohmann::json& resetToDefaults = 1821*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Actions"]["#Manager.ResetToDefaults"]; 18223e40fc74SGunnar Mills resetToDefaults["target"] = 18233e40fc74SGunnar Mills "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults"; 18243e40fc74SGunnar Mills resetToDefaults["ResetType@Redfish.AllowableValues"] = {"ResetAll"}; 18253e40fc74SGunnar Mills 1826*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["DateTime"] = crow::utility::dateTimeNow(); 1827474bfad5SSantosh Puranik 1828f8c3e6f0SKuiying Wang // Fill in SerialConsole info 1829*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["SerialConsole"]["ServiceEnabled"] = true; 1830*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["SerialConsole"]["MaxConcurrentSessions"] = 15; 1831*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["SerialConsole"]["ConnectTypesSupported"] = { 1832*8d1b46d7Szhanghch05 "IPMI", "SSH"}; 1833ef47bb18SSantosh Puranik #ifdef BMCWEB_ENABLE_KVM 1834f8c3e6f0SKuiying Wang // Fill in GraphicalConsole info 1835*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["GraphicalConsole"]["ServiceEnabled"] = true; 1836*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["GraphicalConsole"]["MaxConcurrentSessions"] = 1837*8d1b46d7Szhanghch05 4; 1838*8d1b46d7Szhanghch05 asyncResp->res 1839*8d1b46d7Szhanghch05 .jsonValue["GraphicalConsole"]["ConnectTypesSupported"] = {"KVMIP"}; 1840ef47bb18SSantosh Puranik #endif // BMCWEB_ENABLE_KVM 1841474bfad5SSantosh Puranik 1842*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Links"]["ManagerForServers@odata.count"] = 1; 1843*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Links"]["ManagerForServers"] = { 1844603a6640SGunnar Mills {{"@odata.id", "/redfish/v1/Systems/system"}}}; 184526f03899SShawn McCarney 1846b49ac873SJames Feist auto health = std::make_shared<HealthPopulate>(asyncResp); 1847b49ac873SJames Feist health->isManagersHealth = true; 1848b49ac873SJames Feist health->populate(); 1849b49ac873SJames Feist 1850f97ddba7SGunnar Mills fw_util::populateFirmwareInformation(asyncResp, fw_util::bmcPurpose, 185172d566d9SGunnar Mills "FirmwareVersion", true); 18520f6b00bdSJames Feist 18534bf2b033SGunnar Mills getLastResetTime(asyncResp); 18544bf2b033SGunnar Mills 185573df0db0SJames Feist auto pids = std::make_shared<GetPIDValues>(asyncResp); 185673df0db0SJames Feist pids->run(); 1857c5d03ff4SJennifer Lee 1858*8d1b46d7Szhanghch05 getMainChassisId( 1859*8d1b46d7Szhanghch05 asyncResp, [](const std::string& chassisId, 1860*8d1b46d7Szhanghch05 const std::shared_ptr<bmcweb::AsyncResp>& aRsp) { 1861*8d1b46d7Szhanghch05 aRsp->res.jsonValue["Links"]["ManagerForChassis@odata.count"] = 1862*8d1b46d7Szhanghch05 1; 1863c5d03ff4SJennifer Lee aRsp->res.jsonValue["Links"]["ManagerForChassis"] = { 1864c5d03ff4SJennifer Lee {{"@odata.id", "/redfish/v1/Chassis/" + chassisId}}}; 18652c0feb00SJason M. Bills aRsp->res.jsonValue["Links"]["ManagerInChassis"] = { 18662c0feb00SJason M. Bills {"@odata.id", "/redfish/v1/Chassis/" + chassisId}}; 1867c5d03ff4SJennifer Lee }); 18680f6b00bdSJames Feist 18690f6b00bdSJames Feist static bool started = false; 18700f6b00bdSJames Feist 18710f6b00bdSJames Feist if (!started) 18720f6b00bdSJames Feist { 18730f6b00bdSJames Feist crow::connections::systemBus->async_method_call( 18740f6b00bdSJames Feist [asyncResp](const boost::system::error_code ec, 18750f6b00bdSJames Feist const std::variant<double>& resp) { 18760f6b00bdSJames Feist if (ec) 18770f6b00bdSJames Feist { 18780f6b00bdSJames Feist BMCWEB_LOG_ERROR << "Error while getting progress"; 18790f6b00bdSJames Feist messages::internalError(asyncResp->res); 18800f6b00bdSJames Feist return; 18810f6b00bdSJames Feist } 18820f6b00bdSJames Feist const double* val = std::get_if<double>(&resp); 18830f6b00bdSJames Feist if (val == nullptr) 18840f6b00bdSJames Feist { 18850f6b00bdSJames Feist BMCWEB_LOG_ERROR 18860f6b00bdSJames Feist << "Invalid response while getting progress"; 18870f6b00bdSJames Feist messages::internalError(asyncResp->res); 18880f6b00bdSJames Feist return; 18890f6b00bdSJames Feist } 18900f6b00bdSJames Feist if (*val < 1.0) 18910f6b00bdSJames Feist { 18920f6b00bdSJames Feist asyncResp->res.jsonValue["Status"]["State"] = 18930f6b00bdSJames Feist "Starting"; 18940f6b00bdSJames Feist started = true; 18950f6b00bdSJames Feist } 18960f6b00bdSJames Feist }, 18970f6b00bdSJames Feist "org.freedesktop.systemd1", "/org/freedesktop/systemd1", 18980f6b00bdSJames Feist "org.freedesktop.DBus.Properties", "Get", 18990f6b00bdSJames Feist "org.freedesktop.systemd1.Manager", "Progress"); 19000f6b00bdSJames Feist } 1901ef6ca6e4SChicago Duan 1902ef6ca6e4SChicago Duan crow::connections::systemBus->async_method_call( 1903ef6ca6e4SChicago Duan [asyncResp]( 1904ef6ca6e4SChicago Duan const boost::system::error_code ec, 1905ef6ca6e4SChicago Duan const std::vector<std::pair< 1906ef6ca6e4SChicago Duan std::string, std::vector<std::pair< 1907ef6ca6e4SChicago Duan std::string, std::vector<std::string>>>>>& 1908ef6ca6e4SChicago Duan subtree) { 1909ef6ca6e4SChicago Duan if (ec) 1910ef6ca6e4SChicago Duan { 1911ef6ca6e4SChicago Duan BMCWEB_LOG_DEBUG << "D-Bus response error on GetSubTree " 1912ef6ca6e4SChicago Duan << ec; 1913ef6ca6e4SChicago Duan return; 1914ef6ca6e4SChicago Duan } 1915ef6ca6e4SChicago Duan if (subtree.size() == 0) 1916ef6ca6e4SChicago Duan { 1917ef6ca6e4SChicago Duan BMCWEB_LOG_DEBUG << "Can't find bmc D-Bus object!"; 1918ef6ca6e4SChicago Duan return; 1919ef6ca6e4SChicago Duan } 1920ef6ca6e4SChicago Duan // Assume only 1 bmc D-Bus object 1921ef6ca6e4SChicago Duan // Throw an error if there is more than 1 1922ef6ca6e4SChicago Duan if (subtree.size() > 1) 1923ef6ca6e4SChicago Duan { 1924ef6ca6e4SChicago Duan BMCWEB_LOG_DEBUG << "Found more than 1 bmc D-Bus object!"; 1925ef6ca6e4SChicago Duan messages::internalError(asyncResp->res); 1926ef6ca6e4SChicago Duan return; 1927ef6ca6e4SChicago Duan } 1928ef6ca6e4SChicago Duan 1929ef6ca6e4SChicago Duan if (subtree[0].first.empty() || subtree[0].second.size() != 1) 1930ef6ca6e4SChicago Duan { 1931ef6ca6e4SChicago Duan BMCWEB_LOG_DEBUG << "Error getting bmc D-Bus object!"; 1932ef6ca6e4SChicago Duan messages::internalError(asyncResp->res); 1933ef6ca6e4SChicago Duan return; 1934ef6ca6e4SChicago Duan } 1935ef6ca6e4SChicago Duan 1936ef6ca6e4SChicago Duan const std::string& path = subtree[0].first; 1937ef6ca6e4SChicago Duan const std::string& connectionName = subtree[0].second[0].first; 1938ef6ca6e4SChicago Duan 1939071d8fdfSSunnySrivastava1984 for (const auto& interfaceName : subtree[0].second[0].second) 1940071d8fdfSSunnySrivastava1984 { 1941071d8fdfSSunnySrivastava1984 if (interfaceName == 1942071d8fdfSSunnySrivastava1984 "xyz.openbmc_project.Inventory.Decorator.Asset") 1943071d8fdfSSunnySrivastava1984 { 1944ef6ca6e4SChicago Duan crow::connections::systemBus->async_method_call( 1945ef6ca6e4SChicago Duan [asyncResp]( 1946ef6ca6e4SChicago Duan const boost::system::error_code ec, 1947071d8fdfSSunnySrivastava1984 const std::vector<std::pair< 1948071d8fdfSSunnySrivastava1984 std::string, std::variant<std::string>>>& 1949ef6ca6e4SChicago Duan propertiesList) { 1950ef6ca6e4SChicago Duan if (ec) 1951ef6ca6e4SChicago Duan { 1952ef6ca6e4SChicago Duan BMCWEB_LOG_DEBUG << "Can't get bmc asset!"; 1953ef6ca6e4SChicago Duan return; 1954ef6ca6e4SChicago Duan } 1955ef6ca6e4SChicago Duan for (const std::pair<std::string, 1956ef6ca6e4SChicago Duan std::variant<std::string>>& 1957ef6ca6e4SChicago Duan property : propertiesList) 1958ef6ca6e4SChicago Duan { 1959071d8fdfSSunnySrivastava1984 const std::string& propertyName = 1960071d8fdfSSunnySrivastava1984 property.first; 1961ef6ca6e4SChicago Duan 1962ef6ca6e4SChicago Duan if ((propertyName == "PartNumber") || 1963ef6ca6e4SChicago Duan (propertyName == "SerialNumber") || 1964071d8fdfSSunnySrivastava1984 (propertyName == "Manufacturer") || 1965071d8fdfSSunnySrivastava1984 (propertyName == "Model") || 1966071d8fdfSSunnySrivastava1984 (propertyName == "SparePartNumber")) 1967ef6ca6e4SChicago Duan { 1968ef6ca6e4SChicago Duan const std::string* value = 1969071d8fdfSSunnySrivastava1984 std::get_if<std::string>( 1970071d8fdfSSunnySrivastava1984 &property.second); 1971ef6ca6e4SChicago Duan if (value == nullptr) 1972ef6ca6e4SChicago Duan { 1973ef6ca6e4SChicago Duan // illegal property 1974071d8fdfSSunnySrivastava1984 messages::internalError( 1975071d8fdfSSunnySrivastava1984 asyncResp->res); 1976ef6ca6e4SChicago Duan continue; 1977ef6ca6e4SChicago Duan } 1978071d8fdfSSunnySrivastava1984 asyncResp->res.jsonValue[propertyName] = 1979071d8fdfSSunnySrivastava1984 *value; 1980ef6ca6e4SChicago Duan } 1981ef6ca6e4SChicago Duan } 1982ef6ca6e4SChicago Duan }, 1983071d8fdfSSunnySrivastava1984 connectionName, path, 1984071d8fdfSSunnySrivastava1984 "org.freedesktop.DBus.Properties", "GetAll", 1985071d8fdfSSunnySrivastava1984 "xyz.openbmc_project.Inventory.Decorator.Asset"); 1986071d8fdfSSunnySrivastava1984 } 1987071d8fdfSSunnySrivastava1984 else if (interfaceName == "xyz.openbmc_project.Inventory." 1988071d8fdfSSunnySrivastava1984 "Decorator.LocationCode") 1989071d8fdfSSunnySrivastava1984 { 1990071d8fdfSSunnySrivastava1984 getLocation(asyncResp, connectionName, path); 1991071d8fdfSSunnySrivastava1984 } 1992071d8fdfSSunnySrivastava1984 } 1993ef6ca6e4SChicago Duan }, 1994ef6ca6e4SChicago Duan "xyz.openbmc_project.ObjectMapper", 1995ef6ca6e4SChicago Duan "/xyz/openbmc_project/object_mapper", 1996ef6ca6e4SChicago Duan "xyz.openbmc_project.ObjectMapper", "GetSubTree", 1997ef6ca6e4SChicago Duan "/xyz/openbmc_project/inventory", int32_t(0), 1998ef6ca6e4SChicago Duan std::array<const char*, 1>{ 1999ef6ca6e4SChicago Duan "xyz.openbmc_project.Inventory.Item.Bmc"}); 200083ff9ab6SJames Feist } 20015b4aa86bSJames Feist 2002*8d1b46d7Szhanghch05 void doPatch(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 2003*8d1b46d7Szhanghch05 const crow::Request& req, 2004cb13a392SEd Tanous const std::vector<std::string>&) override 20055b4aa86bSJames Feist { 20060627a2c7SEd Tanous std::optional<nlohmann::json> oem; 20074bfefa74SGunnar Mills std::optional<nlohmann::json> links; 2008af5d6058SSantosh Puranik std::optional<std::string> datetime; 20090627a2c7SEd Tanous 2010*8d1b46d7Szhanghch05 if (!json_util::readJson(req, asyncResp->res, "Oem", oem, "DateTime", 20114bfefa74SGunnar Mills datetime, "Links", links)) 201283ff9ab6SJames Feist { 201383ff9ab6SJames Feist return; 201483ff9ab6SJames Feist } 20150627a2c7SEd Tanous 20160627a2c7SEd Tanous if (oem) 201783ff9ab6SJames Feist { 20185f2caaefSJames Feist std::optional<nlohmann::json> openbmc; 2019*8d1b46d7Szhanghch05 if (!redfish::json_util::readJson(*oem, asyncResp->res, "OpenBmc", 2020*8d1b46d7Szhanghch05 openbmc)) 202183ff9ab6SJames Feist { 202271f52d96SEd Tanous BMCWEB_LOG_ERROR 202371f52d96SEd Tanous << "Illegal Property " 202471f52d96SEd Tanous << oem->dump(2, ' ', true, 202571f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 202683ff9ab6SJames Feist return; 202783ff9ab6SJames Feist } 20285f2caaefSJames Feist if (openbmc) 202983ff9ab6SJames Feist { 20305f2caaefSJames Feist std::optional<nlohmann::json> fan; 2031*8d1b46d7Szhanghch05 if (!redfish::json_util::readJson(*openbmc, asyncResp->res, 2032*8d1b46d7Szhanghch05 "Fan", fan)) 203383ff9ab6SJames Feist { 203471f52d96SEd Tanous BMCWEB_LOG_ERROR 203571f52d96SEd Tanous << "Illegal Property " 203671f52d96SEd Tanous << openbmc->dump( 203771f52d96SEd Tanous 2, ' ', true, 203871f52d96SEd Tanous nlohmann::json::error_handler_t::replace); 203983ff9ab6SJames Feist return; 204083ff9ab6SJames Feist } 20415f2caaefSJames Feist if (fan) 204283ff9ab6SJames Feist { 2043*8d1b46d7Szhanghch05 auto pid = std::make_shared<SetPIDValues>(asyncResp, *fan); 204473df0db0SJames Feist pid->run(); 204583ff9ab6SJames Feist } 204683ff9ab6SJames Feist } 204783ff9ab6SJames Feist } 20484bfefa74SGunnar Mills if (links) 20494bfefa74SGunnar Mills { 20504bfefa74SGunnar Mills std::optional<nlohmann::json> activeSoftwareImage; 2051*8d1b46d7Szhanghch05 if (!redfish::json_util::readJson(*links, asyncResp->res, 2052*8d1b46d7Szhanghch05 "ActiveSoftwareImage", 2053*8d1b46d7Szhanghch05 activeSoftwareImage)) 20544bfefa74SGunnar Mills { 20554bfefa74SGunnar Mills return; 20564bfefa74SGunnar Mills } 20574bfefa74SGunnar Mills if (activeSoftwareImage) 20584bfefa74SGunnar Mills { 20594bfefa74SGunnar Mills std::optional<std::string> odataId; 2060*8d1b46d7Szhanghch05 if (!json_util::readJson(*activeSoftwareImage, asyncResp->res, 2061*8d1b46d7Szhanghch05 "@odata.id", odataId)) 20624bfefa74SGunnar Mills { 20634bfefa74SGunnar Mills return; 20644bfefa74SGunnar Mills } 20654bfefa74SGunnar Mills 20664bfefa74SGunnar Mills if (odataId) 20674bfefa74SGunnar Mills { 2068*8d1b46d7Szhanghch05 setActiveFirmwareImage(asyncResp, *odataId); 20694bfefa74SGunnar Mills } 20704bfefa74SGunnar Mills } 20714bfefa74SGunnar Mills } 2072af5d6058SSantosh Puranik if (datetime) 2073af5d6058SSantosh Puranik { 2074*8d1b46d7Szhanghch05 setDateTime(asyncResp, std::move(*datetime)); 2075af5d6058SSantosh Puranik } 2076af5d6058SSantosh Puranik } 2077af5d6058SSantosh Puranik 2078*8d1b46d7Szhanghch05 void getLastResetTime(const std::shared_ptr<bmcweb::AsyncResp>& aResp) 20794bf2b033SGunnar Mills { 20804bf2b033SGunnar Mills BMCWEB_LOG_DEBUG << "Getting Manager Last Reset Time"; 20814bf2b033SGunnar Mills 20824bf2b033SGunnar Mills crow::connections::systemBus->async_method_call( 20834bf2b033SGunnar Mills [aResp](const boost::system::error_code ec, 20844bf2b033SGunnar Mills std::variant<uint64_t>& lastResetTime) { 20854bf2b033SGunnar Mills if (ec) 20864bf2b033SGunnar Mills { 20874bf2b033SGunnar Mills BMCWEB_LOG_DEBUG << "D-BUS response error " << ec; 20884bf2b033SGunnar Mills return; 20894bf2b033SGunnar Mills } 20904bf2b033SGunnar Mills 20914bf2b033SGunnar Mills const uint64_t* lastResetTimePtr = 20924bf2b033SGunnar Mills std::get_if<uint64_t>(&lastResetTime); 20934bf2b033SGunnar Mills 20944bf2b033SGunnar Mills if (!lastResetTimePtr) 20954bf2b033SGunnar Mills { 20964bf2b033SGunnar Mills messages::internalError(aResp->res); 20974bf2b033SGunnar Mills return; 20984bf2b033SGunnar Mills } 20994bf2b033SGunnar Mills // LastRebootTime is epoch time, in milliseconds 21004bf2b033SGunnar Mills // https://github.com/openbmc/phosphor-dbus-interfaces/blob/7f9a128eb9296e926422ddc312c148b625890bb6/xyz/openbmc_project/State/BMC.interface.yaml#L19 21014bf2b033SGunnar Mills time_t lastResetTimeStamp = 21024bf2b033SGunnar Mills static_cast<time_t>(*lastResetTimePtr / 1000); 21034bf2b033SGunnar Mills 21044bf2b033SGunnar Mills // Convert to ISO 8601 standard 21054bf2b033SGunnar Mills aResp->res.jsonValue["LastResetTime"] = 21064bf2b033SGunnar Mills crow::utility::getDateTime(lastResetTimeStamp); 21074bf2b033SGunnar Mills }, 21084bf2b033SGunnar Mills "xyz.openbmc_project.State.BMC", "/xyz/openbmc_project/state/bmc0", 21094bf2b033SGunnar Mills "org.freedesktop.DBus.Properties", "Get", 21104bf2b033SGunnar Mills "xyz.openbmc_project.State.BMC", "LastRebootTime"); 21114bf2b033SGunnar Mills } 21124bf2b033SGunnar Mills 21134bfefa74SGunnar Mills /** 21144bfefa74SGunnar Mills * @brief Set the running firmware image 21154bfefa74SGunnar Mills * 21164bfefa74SGunnar Mills * @param[i,o] aResp - Async response object 21174bfefa74SGunnar Mills * @param[i] runningFirmwareTarget - Image to make the running image 21184bfefa74SGunnar Mills * 21194bfefa74SGunnar Mills * @return void 21204bfefa74SGunnar Mills */ 2121*8d1b46d7Szhanghch05 void setActiveFirmwareImage(const std::shared_ptr<bmcweb::AsyncResp>& aResp, 2122f23b7296SEd Tanous const std::string& runningFirmwareTarget) 21234bfefa74SGunnar Mills { 21244bfefa74SGunnar Mills // Get the Id from /redfish/v1/UpdateService/FirmwareInventory/<Id> 2125f23b7296SEd Tanous std::string::size_type idPos = runningFirmwareTarget.rfind('/'); 21264bfefa74SGunnar Mills if (idPos == std::string::npos) 21274bfefa74SGunnar Mills { 21284bfefa74SGunnar Mills messages::propertyValueNotInList(aResp->res, runningFirmwareTarget, 21294bfefa74SGunnar Mills "@odata.id"); 21304bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Can't parse firmware ID!"; 21314bfefa74SGunnar Mills return; 21324bfefa74SGunnar Mills } 21334bfefa74SGunnar Mills idPos++; 21344bfefa74SGunnar Mills if (idPos >= runningFirmwareTarget.size()) 21354bfefa74SGunnar Mills { 21364bfefa74SGunnar Mills messages::propertyValueNotInList(aResp->res, runningFirmwareTarget, 21374bfefa74SGunnar Mills "@odata.id"); 21384bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Invalid firmware ID."; 21394bfefa74SGunnar Mills return; 21404bfefa74SGunnar Mills } 21414bfefa74SGunnar Mills std::string firmwareId = runningFirmwareTarget.substr(idPos); 21424bfefa74SGunnar Mills 21434bfefa74SGunnar Mills // Make sure the image is valid before setting priority 21444bfefa74SGunnar Mills crow::connections::systemBus->async_method_call( 21454bfefa74SGunnar Mills [aResp, firmwareId, 21464bfefa74SGunnar Mills runningFirmwareTarget](const boost::system::error_code ec, 21474bfefa74SGunnar Mills ManagedObjectType& subtree) { 21484bfefa74SGunnar Mills if (ec) 21494bfefa74SGunnar Mills { 21504bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "D-Bus response error getting objects."; 21514bfefa74SGunnar Mills messages::internalError(aResp->res); 21524bfefa74SGunnar Mills return; 21534bfefa74SGunnar Mills } 21544bfefa74SGunnar Mills 21554bfefa74SGunnar Mills if (subtree.size() == 0) 21564bfefa74SGunnar Mills { 21574bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Can't find image!"; 21584bfefa74SGunnar Mills messages::internalError(aResp->res); 21594bfefa74SGunnar Mills return; 21604bfefa74SGunnar Mills } 21614bfefa74SGunnar Mills 21624bfefa74SGunnar Mills bool foundImage = false; 21634bfefa74SGunnar Mills for (auto& object : subtree) 21644bfefa74SGunnar Mills { 21654bfefa74SGunnar Mills const std::string& path = 21664bfefa74SGunnar Mills static_cast<const std::string&>(object.first); 2167f23b7296SEd Tanous std::size_t idPos2 = path.rfind('/'); 21684bfefa74SGunnar Mills 21694bfefa74SGunnar Mills if (idPos2 == std::string::npos) 21704bfefa74SGunnar Mills { 21714bfefa74SGunnar Mills continue; 21724bfefa74SGunnar Mills } 21734bfefa74SGunnar Mills 21744bfefa74SGunnar Mills idPos2++; 21754bfefa74SGunnar Mills if (idPos2 >= path.size()) 21764bfefa74SGunnar Mills { 21774bfefa74SGunnar Mills continue; 21784bfefa74SGunnar Mills } 21794bfefa74SGunnar Mills 21804bfefa74SGunnar Mills if (path.substr(idPos2) == firmwareId) 21814bfefa74SGunnar Mills { 21824bfefa74SGunnar Mills foundImage = true; 21834bfefa74SGunnar Mills break; 21844bfefa74SGunnar Mills } 21854bfefa74SGunnar Mills } 21864bfefa74SGunnar Mills 21874bfefa74SGunnar Mills if (!foundImage) 21884bfefa74SGunnar Mills { 21894bfefa74SGunnar Mills messages::propertyValueNotInList( 21904bfefa74SGunnar Mills aResp->res, runningFirmwareTarget, "@odata.id"); 21914bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Invalid firmware ID."; 21924bfefa74SGunnar Mills return; 21934bfefa74SGunnar Mills } 21944bfefa74SGunnar Mills 21954bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "Setting firmware version " + firmwareId + 21964bfefa74SGunnar Mills " to priority 0."; 21974bfefa74SGunnar Mills 21984bfefa74SGunnar Mills // Only support Immediate 21994bfefa74SGunnar Mills // An addition could be a Redfish Setting like 22004bfefa74SGunnar Mills // ActiveSoftwareImageApplyTime and support OnReset 22014bfefa74SGunnar Mills crow::connections::systemBus->async_method_call( 22024bfefa74SGunnar Mills [aResp](const boost::system::error_code ec) { 22034bfefa74SGunnar Mills if (ec) 22044bfefa74SGunnar Mills { 22054bfefa74SGunnar Mills BMCWEB_LOG_DEBUG << "D-Bus response error setting."; 22064bfefa74SGunnar Mills messages::internalError(aResp->res); 22074bfefa74SGunnar Mills return; 22084bfefa74SGunnar Mills } 22094bfefa74SGunnar Mills doBMCGracefulRestart(aResp); 22104bfefa74SGunnar Mills }, 22114bfefa74SGunnar Mills 22124bfefa74SGunnar Mills "xyz.openbmc_project.Software.BMC.Updater", 22134bfefa74SGunnar Mills "/xyz/openbmc_project/software/" + firmwareId, 22144bfefa74SGunnar Mills "org.freedesktop.DBus.Properties", "Set", 22154bfefa74SGunnar Mills "xyz.openbmc_project.Software.RedundancyPriority", 22164bfefa74SGunnar Mills "Priority", std::variant<uint8_t>(static_cast<uint8_t>(0))); 22174bfefa74SGunnar Mills }, 22184bfefa74SGunnar Mills "xyz.openbmc_project.Software.BMC.Updater", 22194bfefa74SGunnar Mills "/xyz/openbmc_project/software", 22204bfefa74SGunnar Mills "org.freedesktop.DBus.ObjectManager", "GetManagedObjects"); 22214bfefa74SGunnar Mills } 22224bfefa74SGunnar Mills 2223*8d1b46d7Szhanghch05 void setDateTime(std::shared_ptr<bmcweb::AsyncResp> aResp, 2224af5d6058SSantosh Puranik std::string datetime) const 2225af5d6058SSantosh Puranik { 2226af5d6058SSantosh Puranik BMCWEB_LOG_DEBUG << "Set date time: " << datetime; 2227af5d6058SSantosh Puranik 2228af5d6058SSantosh Puranik std::stringstream stream(datetime); 2229af5d6058SSantosh Puranik // Convert from ISO 8601 to boost local_time 2230af5d6058SSantosh Puranik // (BMC only has time in UTC) 2231af5d6058SSantosh Puranik boost::posix_time::ptime posixTime; 2232af5d6058SSantosh Puranik boost::posix_time::ptime epoch(boost::gregorian::date(1970, 1, 1)); 2233af5d6058SSantosh Puranik // Facet gets deleted with the stringsteam 2234af5d6058SSantosh Puranik auto ifc = std::make_unique<boost::local_time::local_time_input_facet>( 2235af5d6058SSantosh Puranik "%Y-%m-%d %H:%M:%S%F %ZP"); 2236af5d6058SSantosh Puranik stream.imbue(std::locale(stream.getloc(), ifc.release())); 2237af5d6058SSantosh Puranik 2238af5d6058SSantosh Puranik boost::local_time::local_date_time ldt( 2239af5d6058SSantosh Puranik boost::local_time::not_a_date_time); 2240af5d6058SSantosh Puranik 2241af5d6058SSantosh Puranik if (stream >> ldt) 2242af5d6058SSantosh Puranik { 2243af5d6058SSantosh Puranik posixTime = ldt.utc_time(); 2244af5d6058SSantosh Puranik boost::posix_time::time_duration dur = posixTime - epoch; 2245af5d6058SSantosh Puranik uint64_t durMicroSecs = 2246af5d6058SSantosh Puranik static_cast<uint64_t>(dur.total_microseconds()); 2247af5d6058SSantosh Puranik crow::connections::systemBus->async_method_call( 2248af5d6058SSantosh Puranik [aResp{std::move(aResp)}, datetime{std::move(datetime)}]( 2249af5d6058SSantosh Puranik const boost::system::error_code ec) { 2250af5d6058SSantosh Puranik if (ec) 2251af5d6058SSantosh Puranik { 2252af5d6058SSantosh Puranik BMCWEB_LOG_DEBUG << "Failed to set elapsed time. " 2253af5d6058SSantosh Puranik "DBUS response error " 2254af5d6058SSantosh Puranik << ec; 2255af5d6058SSantosh Puranik messages::internalError(aResp->res); 2256af5d6058SSantosh Puranik return; 2257af5d6058SSantosh Puranik } 2258af5d6058SSantosh Puranik aResp->res.jsonValue["DateTime"] = datetime; 2259af5d6058SSantosh Puranik }, 2260af5d6058SSantosh Puranik "xyz.openbmc_project.Time.Manager", 2261af5d6058SSantosh Puranik "/xyz/openbmc_project/time/bmc", 2262af5d6058SSantosh Puranik "org.freedesktop.DBus.Properties", "Set", 2263af5d6058SSantosh Puranik "xyz.openbmc_project.Time.EpochTime", "Elapsed", 2264af5d6058SSantosh Puranik std::variant<uint64_t>(durMicroSecs)); 2265af5d6058SSantosh Puranik } 2266af5d6058SSantosh Puranik else 2267af5d6058SSantosh Puranik { 2268af5d6058SSantosh Puranik messages::propertyValueFormatError(aResp->res, datetime, 2269af5d6058SSantosh Puranik "DateTime"); 2270af5d6058SSantosh Puranik return; 2271af5d6058SSantosh Puranik } 227283ff9ab6SJames Feist } 22739c310685SBorawski.Lukasz 22740f74e643SEd Tanous std::string uuid; 22759c310685SBorawski.Lukasz }; 22769c310685SBorawski.Lukasz 22771abe55efSEd Tanous class ManagerCollection : public Node 22781abe55efSEd Tanous { 22799c310685SBorawski.Lukasz public: 228052cc112dSEd Tanous ManagerCollection(App& app) : Node(app, "/redfish/v1/Managers/") 22811abe55efSEd Tanous { 2282a434f2bdSEd Tanous entityPrivileges = { 2283a434f2bdSEd Tanous {boost::beast::http::verb::get, {{"Login"}}}, 2284e0d918bcSEd Tanous {boost::beast::http::verb::head, {{"Login"}}}, 2285e0d918bcSEd Tanous {boost::beast::http::verb::patch, {{"ConfigureManager"}}}, 2286e0d918bcSEd Tanous {boost::beast::http::verb::put, {{"ConfigureManager"}}}, 2287e0d918bcSEd Tanous {boost::beast::http::verb::delete_, {{"ConfigureManager"}}}, 2288e0d918bcSEd Tanous {boost::beast::http::verb::post, {{"ConfigureManager"}}}}; 22899c310685SBorawski.Lukasz } 22909c310685SBorawski.Lukasz 22919c310685SBorawski.Lukasz private: 2292*8d1b46d7Szhanghch05 void doGet(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 2293*8d1b46d7Szhanghch05 const crow::Request&, const std::vector<std::string>&) override 22941abe55efSEd Tanous { 229583ff9ab6SJames Feist // Collections don't include the static data added by SubRoute 229683ff9ab6SJames Feist // because it has a duplicate entry for members 2297*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers"; 2298*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["@odata.type"] = 2299*8d1b46d7Szhanghch05 "#ManagerCollection.ManagerCollection"; 2300*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Name"] = "Manager Collection"; 2301*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Members@odata.count"] = 1; 2302*8d1b46d7Szhanghch05 asyncResp->res.jsonValue["Members"] = { 23035b4aa86bSJames Feist {{"@odata.id", "/redfish/v1/Managers/bmc"}}}; 23049c310685SBorawski.Lukasz } 23059c310685SBorawski.Lukasz }; 23069c310685SBorawski.Lukasz } // namespace redfish 2307