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 189c310685SBorawski.Lukasz #include "node.hpp" 199c310685SBorawski.Lukasz 205b4aa86bSJames Feist #include <boost/algorithm/string/replace.hpp> 21af5d6058SSantosh Puranik #include <boost/date_time.hpp> 225b4aa86bSJames Feist #include <dbus_utility.hpp> 23*73df0db0SJames Feist #include <memory> 24af5d6058SSantosh Puranik #include <sstream> 257bffdb7eSBernard Wong #include <utils/systemd_utils.hpp> 26abf2add6SEd Tanous #include <variant> 275b4aa86bSJames Feist 281abe55efSEd Tanous namespace redfish 291abe55efSEd Tanous { 30ed5befbdSJennifer Lee 31ed5befbdSJennifer Lee /** 32ed5befbdSJennifer Lee * ManagerActionsReset class supports handle POST method for Reset action. 33ed5befbdSJennifer Lee * The class retrieves and sends data directly to dbus. 34ed5befbdSJennifer Lee */ 35ed5befbdSJennifer Lee class ManagerActionsReset : public Node 36ed5befbdSJennifer Lee { 37ed5befbdSJennifer Lee public: 38ed5befbdSJennifer Lee ManagerActionsReset(CrowApp& app) : 39ed5befbdSJennifer Lee Node(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/") 40ed5befbdSJennifer Lee { 41ed5befbdSJennifer Lee entityPrivileges = { 42ed5befbdSJennifer Lee {boost::beast::http::verb::get, {{"Login"}}}, 43ed5befbdSJennifer Lee {boost::beast::http::verb::head, {{"Login"}}}, 44ed5befbdSJennifer Lee {boost::beast::http::verb::patch, {{"ConfigureManager"}}}, 45ed5befbdSJennifer Lee {boost::beast::http::verb::put, {{"ConfigureManager"}}}, 46ed5befbdSJennifer Lee {boost::beast::http::verb::delete_, {{"ConfigureManager"}}}, 47ed5befbdSJennifer Lee {boost::beast::http::verb::post, {{"ConfigureManager"}}}}; 48ed5befbdSJennifer Lee } 49ed5befbdSJennifer Lee 50ed5befbdSJennifer Lee private: 51ed5befbdSJennifer Lee /** 52ed5befbdSJennifer Lee * Function handles POST method request. 53ed5befbdSJennifer Lee * Analyzes POST body message before sends Reset request data to dbus. 54ed5befbdSJennifer Lee * OpenBMC allows for ResetType is GracefulRestart only. 55ed5befbdSJennifer Lee */ 56ed5befbdSJennifer Lee void doPost(crow::Response& res, const crow::Request& req, 57ed5befbdSJennifer Lee const std::vector<std::string>& params) override 58ed5befbdSJennifer Lee { 59ed5befbdSJennifer Lee std::string resetType; 60ed5befbdSJennifer Lee 61ed5befbdSJennifer Lee if (!json_util::readJson(req, res, "ResetType", resetType)) 62ed5befbdSJennifer Lee { 63ed5befbdSJennifer Lee return; 64ed5befbdSJennifer Lee } 65ed5befbdSJennifer Lee 66ed5befbdSJennifer Lee if (resetType != "GracefulRestart") 67ed5befbdSJennifer Lee { 68ed5befbdSJennifer Lee res.result(boost::beast::http::status::bad_request); 69ed5befbdSJennifer Lee messages::actionParameterNotSupported(res, resetType, "ResetType"); 70ed5befbdSJennifer Lee BMCWEB_LOG_ERROR << "Request incorrect action parameter: " 71ed5befbdSJennifer Lee << resetType; 72ed5befbdSJennifer Lee res.end(); 73ed5befbdSJennifer Lee return; 74ed5befbdSJennifer Lee } 75ed5befbdSJennifer Lee doBMCGracefulRestart(res, req, params); 76ed5befbdSJennifer Lee } 77ed5befbdSJennifer Lee 78ed5befbdSJennifer Lee /** 79ed5befbdSJennifer Lee * Function transceives data with dbus directly. 80ed5befbdSJennifer Lee * All BMC state properties will be retrieved before sending reset request. 81ed5befbdSJennifer Lee */ 82ed5befbdSJennifer Lee void doBMCGracefulRestart(crow::Response& res, const crow::Request& req, 83ed5befbdSJennifer Lee const std::vector<std::string>& params) 84ed5befbdSJennifer Lee { 85ed5befbdSJennifer Lee const char* processName = "xyz.openbmc_project.State.BMC"; 86ed5befbdSJennifer Lee const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 87ed5befbdSJennifer Lee const char* interfaceName = "xyz.openbmc_project.State.BMC"; 88ed5befbdSJennifer Lee const std::string& propertyValue = 89ed5befbdSJennifer Lee "xyz.openbmc_project.State.BMC.Transition.Reboot"; 90ed5befbdSJennifer Lee const char* destProperty = "RequestedBMCTransition"; 91ed5befbdSJennifer Lee 92ed5befbdSJennifer Lee // Create the D-Bus variant for D-Bus call. 93ed5befbdSJennifer Lee VariantType dbusPropertyValue(propertyValue); 94ed5befbdSJennifer Lee 95ed5befbdSJennifer Lee std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res); 96ed5befbdSJennifer Lee 97ed5befbdSJennifer Lee crow::connections::systemBus->async_method_call( 98ed5befbdSJennifer Lee [asyncResp](const boost::system::error_code ec) { 99ed5befbdSJennifer Lee // Use "Set" method to set the property value. 100ed5befbdSJennifer Lee if (ec) 101ed5befbdSJennifer Lee { 102ed5befbdSJennifer Lee BMCWEB_LOG_ERROR << "[Set] Bad D-Bus request error: " << ec; 103ed5befbdSJennifer Lee messages::internalError(asyncResp->res); 104ed5befbdSJennifer Lee return; 105ed5befbdSJennifer Lee } 106ed5befbdSJennifer Lee 107ed5befbdSJennifer Lee messages::success(asyncResp->res); 108ed5befbdSJennifer Lee }, 109ed5befbdSJennifer Lee processName, objectPath, "org.freedesktop.DBus.Properties", "Set", 110ed5befbdSJennifer Lee interfaceName, destProperty, dbusPropertyValue); 111ed5befbdSJennifer Lee } 112ed5befbdSJennifer Lee }; 113ed5befbdSJennifer Lee 1145b4aa86bSJames Feist static constexpr const char* objectManagerIface = 1155b4aa86bSJames Feist "org.freedesktop.DBus.ObjectManager"; 1165b4aa86bSJames Feist static constexpr const char* pidConfigurationIface = 1175b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid"; 1185b4aa86bSJames Feist static constexpr const char* pidZoneConfigurationIface = 1195b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid.Zone"; 120b7a08d04SJames Feist static constexpr const char* stepwiseConfigurationIface = 121b7a08d04SJames Feist "xyz.openbmc_project.Configuration.Stepwise"; 122*73df0db0SJames Feist static constexpr const char* thermalModeIface = 123*73df0db0SJames Feist "xyz.openbmc_project.Control.ThermalMode"; 1249c310685SBorawski.Lukasz 1255b4aa86bSJames Feist static void asyncPopulatePid(const std::string& connection, 1265b4aa86bSJames Feist const std::string& path, 127*73df0db0SJames Feist const std::string& currentProfile, 128*73df0db0SJames Feist const std::vector<std::string>& supportedProfiles, 1295b4aa86bSJames Feist std::shared_ptr<AsyncResp> asyncResp) 1305b4aa86bSJames Feist { 1315b4aa86bSJames Feist 1325b4aa86bSJames Feist crow::connections::systemBus->async_method_call( 133*73df0db0SJames Feist [asyncResp, currentProfile, supportedProfiles]( 134*73df0db0SJames Feist const boost::system::error_code ec, 1355b4aa86bSJames Feist const dbus::utility::ManagedObjectType& managedObj) { 1365b4aa86bSJames Feist if (ec) 1375b4aa86bSJames Feist { 1385b4aa86bSJames Feist BMCWEB_LOG_ERROR << ec; 1395b4aa86bSJames Feist asyncResp->res.jsonValue.clear(); 140f12894f8SJason M. Bills messages::internalError(asyncResp->res); 1415b4aa86bSJames Feist return; 1425b4aa86bSJames Feist } 1435b4aa86bSJames Feist nlohmann::json& configRoot = 1445b4aa86bSJames Feist asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"]; 1455b4aa86bSJames Feist nlohmann::json& fans = configRoot["FanControllers"]; 1465b4aa86bSJames Feist fans["@odata.type"] = "#OemManager.FanControllers"; 1475b4aa86bSJames Feist fans["@odata.context"] = 1485b4aa86bSJames Feist "/redfish/v1/$metadata#OemManager.FanControllers"; 1495b4aa86bSJames Feist fans["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/" 1505b4aa86bSJames Feist "Fan/FanControllers"; 1515b4aa86bSJames Feist 1525b4aa86bSJames Feist nlohmann::json& pids = configRoot["PidControllers"]; 1535b4aa86bSJames Feist pids["@odata.type"] = "#OemManager.PidControllers"; 1545b4aa86bSJames Feist pids["@odata.context"] = 1555b4aa86bSJames Feist "/redfish/v1/$metadata#OemManager.PidControllers"; 1565b4aa86bSJames Feist pids["@odata.id"] = 1575b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers"; 1585b4aa86bSJames Feist 159b7a08d04SJames Feist nlohmann::json& stepwise = configRoot["StepwiseControllers"]; 160b7a08d04SJames Feist stepwise["@odata.type"] = "#OemManager.StepwiseControllers"; 161b7a08d04SJames Feist stepwise["@odata.context"] = 162b7a08d04SJames Feist "/redfish/v1/$metadata#OemManager.StepwiseControllers"; 163b7a08d04SJames Feist stepwise["@odata.id"] = 164b7a08d04SJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers"; 165b7a08d04SJames Feist 1665b4aa86bSJames Feist nlohmann::json& zones = configRoot["FanZones"]; 1675b4aa86bSJames Feist zones["@odata.id"] = 1685b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones"; 1695b4aa86bSJames Feist zones["@odata.type"] = "#OemManager.FanZones"; 1705b4aa86bSJames Feist zones["@odata.context"] = 1715b4aa86bSJames Feist "/redfish/v1/$metadata#OemManager.FanZones"; 1725b4aa86bSJames Feist configRoot["@odata.id"] = 1735b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan"; 1745b4aa86bSJames Feist configRoot["@odata.type"] = "#OemManager.Fan"; 1755b4aa86bSJames Feist configRoot["@odata.context"] = 1765b4aa86bSJames Feist "/redfish/v1/$metadata#OemManager.Fan"; 177*73df0db0SJames Feist configRoot["Profile@Redfish.AllowableValues"] = supportedProfiles; 178*73df0db0SJames Feist 179*73df0db0SJames Feist if (!currentProfile.empty()) 180*73df0db0SJames Feist { 181*73df0db0SJames Feist configRoot["Profile"] = currentProfile; 182*73df0db0SJames Feist } 183*73df0db0SJames Feist BMCWEB_LOG_ERROR << "profile = " << currentProfile << " !"; 1845b4aa86bSJames Feist 1855b4aa86bSJames Feist for (const auto& pathPair : managedObj) 1865b4aa86bSJames Feist { 1875b4aa86bSJames Feist for (const auto& intfPair : pathPair.second) 1885b4aa86bSJames Feist { 1895b4aa86bSJames Feist if (intfPair.first != pidConfigurationIface && 190b7a08d04SJames Feist intfPair.first != pidZoneConfigurationIface && 191b7a08d04SJames Feist intfPair.first != stepwiseConfigurationIface) 1925b4aa86bSJames Feist { 1935b4aa86bSJames Feist continue; 1945b4aa86bSJames Feist } 1955b4aa86bSJames Feist auto findName = intfPair.second.find("Name"); 1965b4aa86bSJames Feist if (findName == intfPair.second.end()) 1975b4aa86bSJames Feist { 1985b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Field missing Name"; 199a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 2005b4aa86bSJames Feist return; 2015b4aa86bSJames Feist } 202*73df0db0SJames Feist 2035b4aa86bSJames Feist const std::string* namePtr = 204abf2add6SEd Tanous std::get_if<std::string>(&findName->second); 2055b4aa86bSJames Feist if (namePtr == nullptr) 2065b4aa86bSJames Feist { 2075b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Name Field illegal"; 208b7a08d04SJames Feist messages::internalError(asyncResp->res); 2095b4aa86bSJames Feist return; 2105b4aa86bSJames Feist } 2115b4aa86bSJames Feist std::string name = *namePtr; 2125b4aa86bSJames Feist dbus::utility::escapePathForDbus(name); 213*73df0db0SJames Feist 214*73df0db0SJames Feist auto findProfiles = intfPair.second.find("Profiles"); 215*73df0db0SJames Feist if (findProfiles != intfPair.second.end()) 216*73df0db0SJames Feist { 217*73df0db0SJames Feist const std::vector<std::string>* profiles = 218*73df0db0SJames Feist std::get_if<std::vector<std::string>>( 219*73df0db0SJames Feist &findProfiles->second); 220*73df0db0SJames Feist if (profiles == nullptr) 221*73df0db0SJames Feist { 222*73df0db0SJames Feist BMCWEB_LOG_ERROR << "Pid Profiles Field illegal"; 223*73df0db0SJames Feist messages::internalError(asyncResp->res); 224*73df0db0SJames Feist return; 225*73df0db0SJames Feist } 226*73df0db0SJames Feist if (std::find(profiles->begin(), profiles->end(), 227*73df0db0SJames Feist currentProfile) == profiles->end()) 228*73df0db0SJames Feist { 229*73df0db0SJames Feist BMCWEB_LOG_INFO 230*73df0db0SJames Feist << name << " not supported in current profile"; 231*73df0db0SJames Feist continue; 232*73df0db0SJames Feist } 233*73df0db0SJames Feist } 234b7a08d04SJames Feist nlohmann::json* config = nullptr; 235c33a90ecSJames Feist 236c33a90ecSJames Feist const std::string* classPtr = nullptr; 237c33a90ecSJames Feist auto findClass = intfPair.second.find("Class"); 238c33a90ecSJames Feist if (findClass != intfPair.second.end()) 239c33a90ecSJames Feist { 240c33a90ecSJames Feist classPtr = std::get_if<std::string>(&findClass->second); 241c33a90ecSJames Feist } 242c33a90ecSJames Feist 2435b4aa86bSJames Feist if (intfPair.first == pidZoneConfigurationIface) 2445b4aa86bSJames Feist { 2455b4aa86bSJames Feist std::string chassis; 2465b4aa86bSJames Feist if (!dbus::utility::getNthStringFromPath( 2475b4aa86bSJames Feist pathPair.first.str, 5, chassis)) 2485b4aa86bSJames Feist { 2495b4aa86bSJames Feist chassis = "#IllegalValue"; 2505b4aa86bSJames Feist } 2515b4aa86bSJames Feist nlohmann::json& zone = zones[name]; 2525b4aa86bSJames Feist zone["Chassis"] = { 2535b4aa86bSJames Feist {"@odata.id", "/redfish/v1/Chassis/" + chassis}}; 2545b4aa86bSJames Feist zone["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/" 2555b4aa86bSJames Feist "OpenBmc/Fan/FanZones/" + 2565b4aa86bSJames Feist name; 2575b4aa86bSJames Feist zone["@odata.type"] = "#OemManager.FanZone"; 2585b4aa86bSJames Feist zone["@odata.context"] = 2595b4aa86bSJames Feist "/redfish/v1/$metadata#OemManager.FanZone"; 260b7a08d04SJames Feist config = &zone; 2615b4aa86bSJames Feist } 2625b4aa86bSJames Feist 263b7a08d04SJames Feist else if (intfPair.first == stepwiseConfigurationIface) 2645b4aa86bSJames Feist { 265c33a90ecSJames Feist if (classPtr == nullptr) 266c33a90ecSJames Feist { 267c33a90ecSJames Feist BMCWEB_LOG_ERROR << "Pid Class Field illegal"; 268c33a90ecSJames Feist messages::internalError(asyncResp->res); 269c33a90ecSJames Feist return; 270c33a90ecSJames Feist } 271c33a90ecSJames Feist 272b7a08d04SJames Feist nlohmann::json& controller = stepwise[name]; 273b7a08d04SJames Feist config = &controller; 2745b4aa86bSJames Feist 275b7a08d04SJames Feist controller["@odata.id"] = 276b7a08d04SJames Feist "/redfish/v1/Managers/bmc#/Oem/" 277b7a08d04SJames Feist "OpenBmc/Fan/StepwiseControllers/" + 278b7a08d04SJames Feist std::string(name); 279b7a08d04SJames Feist controller["@odata.type"] = 280b7a08d04SJames Feist "#OemManager.StepwiseController"; 281b7a08d04SJames Feist 282b7a08d04SJames Feist controller["@odata.context"] = 283b7a08d04SJames Feist "/redfish/v1/" 284b7a08d04SJames Feist "$metadata#OemManager.StepwiseController"; 285c33a90ecSJames Feist controller["Direction"] = *classPtr; 2865b4aa86bSJames Feist } 2875b4aa86bSJames Feist 2885b4aa86bSJames Feist // pid and fans are off the same configuration 289b7a08d04SJames Feist else if (intfPair.first == pidConfigurationIface) 2905b4aa86bSJames Feist { 291c33a90ecSJames Feist 2925b4aa86bSJames Feist if (classPtr == nullptr) 2935b4aa86bSJames Feist { 2945b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Class Field illegal"; 295a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 2965b4aa86bSJames Feist return; 2975b4aa86bSJames Feist } 2985b4aa86bSJames Feist bool isFan = *classPtr == "fan"; 2995b4aa86bSJames Feist nlohmann::json& element = 3005b4aa86bSJames Feist isFan ? fans[name] : pids[name]; 301b7a08d04SJames Feist config = &element; 3025b4aa86bSJames Feist if (isFan) 3035b4aa86bSJames Feist { 3045b4aa86bSJames Feist element["@odata.id"] = 3055b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/" 3065b4aa86bSJames Feist "OpenBmc/Fan/FanControllers/" + 3075b4aa86bSJames Feist std::string(name); 3085b4aa86bSJames Feist element["@odata.type"] = 3095b4aa86bSJames Feist "#OemManager.FanController"; 3105b4aa86bSJames Feist 3115b4aa86bSJames Feist element["@odata.context"] = 3125b4aa86bSJames Feist "/redfish/v1/" 3135b4aa86bSJames Feist "$metadata#OemManager.FanController"; 3145b4aa86bSJames Feist } 3155b4aa86bSJames Feist else 3165b4aa86bSJames Feist { 3175b4aa86bSJames Feist element["@odata.id"] = 3185b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/" 3195b4aa86bSJames Feist "OpenBmc/Fan/PidControllers/" + 3205b4aa86bSJames Feist std::string(name); 3215b4aa86bSJames Feist element["@odata.type"] = 3225b4aa86bSJames Feist "#OemManager.PidController"; 3235b4aa86bSJames Feist element["@odata.context"] = 3245b4aa86bSJames Feist "/redfish/v1/$metadata" 3255b4aa86bSJames Feist "#OemManager.PidController"; 3265b4aa86bSJames Feist } 327b7a08d04SJames Feist } 328b7a08d04SJames Feist else 329b7a08d04SJames Feist { 330b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Unexpected configuration"; 331b7a08d04SJames Feist messages::internalError(asyncResp->res); 332b7a08d04SJames Feist return; 333b7a08d04SJames Feist } 334b7a08d04SJames Feist 335b7a08d04SJames Feist // used for making maps out of 2 vectors 336b7a08d04SJames Feist const std::vector<double>* keys = nullptr; 337b7a08d04SJames Feist const std::vector<double>* values = nullptr; 338b7a08d04SJames Feist 339b7a08d04SJames Feist for (const auto& propertyPair : intfPair.second) 340b7a08d04SJames Feist { 341b7a08d04SJames Feist if (propertyPair.first == "Type" || 342b7a08d04SJames Feist propertyPair.first == "Class" || 343b7a08d04SJames Feist propertyPair.first == "Name") 344b7a08d04SJames Feist { 345b7a08d04SJames Feist continue; 346b7a08d04SJames Feist } 347b7a08d04SJames Feist 348b7a08d04SJames Feist // zones 349b7a08d04SJames Feist if (intfPair.first == pidZoneConfigurationIface) 350b7a08d04SJames Feist { 351b7a08d04SJames Feist const double* ptr = 352abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 353b7a08d04SJames Feist if (ptr == nullptr) 354b7a08d04SJames Feist { 355b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 356b7a08d04SJames Feist << propertyPair.first; 357b7a08d04SJames Feist messages::internalError(asyncResp->res); 358b7a08d04SJames Feist return; 359b7a08d04SJames Feist } 360b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 361b7a08d04SJames Feist } 362b7a08d04SJames Feist 363b7a08d04SJames Feist if (intfPair.first == stepwiseConfigurationIface) 364b7a08d04SJames Feist { 365b7a08d04SJames Feist if (propertyPair.first == "Reading" || 366b7a08d04SJames Feist propertyPair.first == "Output") 367b7a08d04SJames Feist { 368b7a08d04SJames Feist const std::vector<double>* ptr = 369abf2add6SEd Tanous std::get_if<std::vector<double>>( 370b7a08d04SJames Feist &propertyPair.second); 371b7a08d04SJames Feist 372b7a08d04SJames Feist if (ptr == nullptr) 373b7a08d04SJames Feist { 374b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 375b7a08d04SJames Feist << propertyPair.first; 376b7a08d04SJames Feist messages::internalError(asyncResp->res); 377b7a08d04SJames Feist return; 378b7a08d04SJames Feist } 379b7a08d04SJames Feist 380b7a08d04SJames Feist if (propertyPair.first == "Reading") 381b7a08d04SJames Feist { 382b7a08d04SJames Feist keys = ptr; 383b7a08d04SJames Feist } 384b7a08d04SJames Feist else 385b7a08d04SJames Feist { 386b7a08d04SJames Feist values = ptr; 387b7a08d04SJames Feist } 388b7a08d04SJames Feist if (keys && values) 389b7a08d04SJames Feist { 390b7a08d04SJames Feist if (keys->size() != values->size()) 391b7a08d04SJames Feist { 392b7a08d04SJames Feist BMCWEB_LOG_ERROR 393b7a08d04SJames Feist << "Reading and Output size don't " 394b7a08d04SJames Feist "match "; 395b7a08d04SJames Feist messages::internalError(asyncResp->res); 396b7a08d04SJames Feist return; 397b7a08d04SJames Feist } 398b7a08d04SJames Feist nlohmann::json& steps = (*config)["Steps"]; 399b7a08d04SJames Feist steps = nlohmann::json::array(); 400b7a08d04SJames Feist for (size_t ii = 0; ii < keys->size(); ii++) 401b7a08d04SJames Feist { 402b7a08d04SJames Feist steps.push_back( 403b7a08d04SJames Feist {{"Target", (*keys)[ii]}, 404b7a08d04SJames Feist {"Output", (*values)[ii]}}); 405b7a08d04SJames Feist } 406b7a08d04SJames Feist } 407b7a08d04SJames Feist } 408b7a08d04SJames Feist if (propertyPair.first == "NegativeHysteresis" || 409b7a08d04SJames Feist propertyPair.first == "PositiveHysteresis") 410b7a08d04SJames Feist { 411b7a08d04SJames Feist const double* ptr = 412abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 413b7a08d04SJames Feist if (ptr == nullptr) 414b7a08d04SJames Feist { 415b7a08d04SJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 416b7a08d04SJames Feist << propertyPair.first; 417b7a08d04SJames Feist messages::internalError(asyncResp->res); 418b7a08d04SJames Feist return; 419b7a08d04SJames Feist } 420b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 421b7a08d04SJames Feist } 422b7a08d04SJames Feist } 423b7a08d04SJames Feist 424b7a08d04SJames Feist // pid and fans are off the same configuration 425b7a08d04SJames Feist if (intfPair.first == pidConfigurationIface || 426b7a08d04SJames Feist intfPair.first == stepwiseConfigurationIface) 427b7a08d04SJames Feist { 4285b4aa86bSJames Feist 4295b4aa86bSJames Feist if (propertyPair.first == "Zones") 4305b4aa86bSJames Feist { 4315b4aa86bSJames Feist const std::vector<std::string>* inputs = 432abf2add6SEd Tanous std::get_if<std::vector<std::string>>( 4331b6b96c5SEd Tanous &propertyPair.second); 4345b4aa86bSJames Feist 4355b4aa86bSJames Feist if (inputs == nullptr) 4365b4aa86bSJames Feist { 4375b4aa86bSJames Feist BMCWEB_LOG_ERROR 4385b4aa86bSJames Feist << "Zones Pid Field Illegal"; 439a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 4405b4aa86bSJames Feist return; 4415b4aa86bSJames Feist } 442b7a08d04SJames Feist auto& data = (*config)[propertyPair.first]; 4435b4aa86bSJames Feist data = nlohmann::json::array(); 4445b4aa86bSJames Feist for (std::string itemCopy : *inputs) 4455b4aa86bSJames Feist { 4465b4aa86bSJames Feist dbus::utility::escapePathForDbus(itemCopy); 4475b4aa86bSJames Feist data.push_back( 4485b4aa86bSJames Feist {{"@odata.id", 4495b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/" 4505b4aa86bSJames Feist "OpenBmc/Fan/FanZones/" + 4515b4aa86bSJames Feist itemCopy}}); 4525b4aa86bSJames Feist } 4535b4aa86bSJames Feist } 4545b4aa86bSJames Feist // todo(james): may never happen, but this 4555b4aa86bSJames Feist // assumes configuration data referenced in the 4565b4aa86bSJames Feist // PID config is provided by the same daemon, we 4575b4aa86bSJames Feist // could add another loop to cover all cases, 4585b4aa86bSJames Feist // but I'm okay kicking this can down the road a 4595b4aa86bSJames Feist // bit 4605b4aa86bSJames Feist 4615b4aa86bSJames Feist else if (propertyPair.first == "Inputs" || 4625b4aa86bSJames Feist propertyPair.first == "Outputs") 4635b4aa86bSJames Feist { 464b7a08d04SJames Feist auto& data = (*config)[propertyPair.first]; 4655b4aa86bSJames Feist const std::vector<std::string>* inputs = 466abf2add6SEd Tanous std::get_if<std::vector<std::string>>( 4671b6b96c5SEd Tanous &propertyPair.second); 4685b4aa86bSJames Feist 4695b4aa86bSJames Feist if (inputs == nullptr) 4705b4aa86bSJames Feist { 4715b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 4725b4aa86bSJames Feist << propertyPair.first; 473f12894f8SJason M. Bills messages::internalError(asyncResp->res); 4745b4aa86bSJames Feist return; 4755b4aa86bSJames Feist } 4765b4aa86bSJames Feist data = *inputs; 4775b4aa86bSJames Feist } // doubles 4785b4aa86bSJames Feist else if (propertyPair.first == 4795b4aa86bSJames Feist "FFGainCoefficient" || 4805b4aa86bSJames Feist propertyPair.first == "FFOffCoefficient" || 4815b4aa86bSJames Feist propertyPair.first == "ICoefficient" || 4825b4aa86bSJames Feist propertyPair.first == "ILimitMax" || 4835b4aa86bSJames Feist propertyPair.first == "ILimitMin" || 484aad1a257SJames Feist propertyPair.first == 485aad1a257SJames Feist "PositiveHysteresis" || 486aad1a257SJames Feist propertyPair.first == 487aad1a257SJames Feist "NegativeHysteresis" || 4885b4aa86bSJames Feist propertyPair.first == "OutLimitMax" || 4895b4aa86bSJames Feist propertyPair.first == "OutLimitMin" || 4905b4aa86bSJames Feist propertyPair.first == "PCoefficient" || 4917625cb81SJames Feist propertyPair.first == "SetPoint" || 4925b4aa86bSJames Feist propertyPair.first == "SlewNeg" || 4935b4aa86bSJames Feist propertyPair.first == "SlewPos") 4945b4aa86bSJames Feist { 4955b4aa86bSJames Feist const double* ptr = 496abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 4975b4aa86bSJames Feist if (ptr == nullptr) 4985b4aa86bSJames Feist { 4995b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 5005b4aa86bSJames Feist << propertyPair.first; 501f12894f8SJason M. Bills messages::internalError(asyncResp->res); 5025b4aa86bSJames Feist return; 5035b4aa86bSJames Feist } 504b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 5055b4aa86bSJames Feist } 5065b4aa86bSJames Feist } 5075b4aa86bSJames Feist } 5085b4aa86bSJames Feist } 5095b4aa86bSJames Feist } 5105b4aa86bSJames Feist }, 5115b4aa86bSJames Feist connection, path, objectManagerIface, "GetManagedObjects"); 5125b4aa86bSJames Feist } 513ca537928SJennifer Lee 51483ff9ab6SJames Feist enum class CreatePIDRet 51583ff9ab6SJames Feist { 51683ff9ab6SJames Feist fail, 51783ff9ab6SJames Feist del, 51883ff9ab6SJames Feist patch 51983ff9ab6SJames Feist }; 52083ff9ab6SJames Feist 5215f2caaefSJames Feist static bool getZonesFromJsonReq(const std::shared_ptr<AsyncResp>& response, 5225f2caaefSJames Feist std::vector<nlohmann::json>& config, 5235f2caaefSJames Feist std::vector<std::string>& zones) 5245f2caaefSJames Feist { 525b6baeaa4SJames Feist if (config.empty()) 526b6baeaa4SJames Feist { 527b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Empty Zones"; 528b6baeaa4SJames Feist messages::propertyValueFormatError(response->res, 529b6baeaa4SJames Feist nlohmann::json::array(), "Zones"); 530b6baeaa4SJames Feist return false; 531b6baeaa4SJames Feist } 5325f2caaefSJames Feist for (auto& odata : config) 5335f2caaefSJames Feist { 5345f2caaefSJames Feist std::string path; 5355f2caaefSJames Feist if (!redfish::json_util::readJson(odata, response->res, "@odata.id", 5365f2caaefSJames Feist path)) 5375f2caaefSJames Feist { 5385f2caaefSJames Feist return false; 5395f2caaefSJames Feist } 5405f2caaefSJames Feist std::string input; 54161adbda3SJames Feist 54261adbda3SJames Feist // 8 below comes from 54361adbda3SJames Feist // /redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/Left 54461adbda3SJames Feist // 0 1 2 3 4 5 6 7 8 54561adbda3SJames Feist if (!dbus::utility::getNthStringFromPath(path, 8, input)) 5465f2caaefSJames Feist { 5475f2caaefSJames Feist BMCWEB_LOG_ERROR << "Got invalid path " << path; 5485f2caaefSJames Feist BMCWEB_LOG_ERROR << "Illegal Type Zones"; 5495f2caaefSJames Feist messages::propertyValueFormatError(response->res, odata.dump(), 5505f2caaefSJames Feist "Zones"); 5515f2caaefSJames Feist return false; 5525f2caaefSJames Feist } 5535f2caaefSJames Feist boost::replace_all(input, "_", " "); 5545f2caaefSJames Feist zones.emplace_back(std::move(input)); 5555f2caaefSJames Feist } 5565f2caaefSJames Feist return true; 5575f2caaefSJames Feist } 5585f2caaefSJames Feist 559*73df0db0SJames Feist static const dbus::utility::ManagedItem* 560*73df0db0SJames Feist findChassis(const dbus::utility::ManagedObjectType& managedObj, 561b6baeaa4SJames Feist const std::string& value, std::string& chassis) 562b6baeaa4SJames Feist { 563b6baeaa4SJames Feist BMCWEB_LOG_DEBUG << "Find Chassis: " << value << "\n"; 564b6baeaa4SJames Feist 565b6baeaa4SJames Feist std::string escaped = boost::replace_all_copy(value, " ", "_"); 566b6baeaa4SJames Feist escaped = "/" + escaped; 567b6baeaa4SJames Feist auto it = std::find_if( 568b6baeaa4SJames Feist managedObj.begin(), managedObj.end(), [&escaped](const auto& obj) { 569b6baeaa4SJames Feist if (boost::algorithm::ends_with(obj.first.str, escaped)) 570b6baeaa4SJames Feist { 571b6baeaa4SJames Feist BMCWEB_LOG_DEBUG << "Matched " << obj.first.str << "\n"; 572b6baeaa4SJames Feist return true; 573b6baeaa4SJames Feist } 574b6baeaa4SJames Feist return false; 575b6baeaa4SJames Feist }); 576b6baeaa4SJames Feist 577b6baeaa4SJames Feist if (it == managedObj.end()) 578b6baeaa4SJames Feist { 579*73df0db0SJames Feist return nullptr; 580b6baeaa4SJames Feist } 581b6baeaa4SJames Feist // 5 comes from <chassis-name> being the 5th element 582b6baeaa4SJames Feist // /xyz/openbmc_project/inventory/system/chassis/<chassis-name> 583*73df0db0SJames Feist if (dbus::utility::getNthStringFromPath(it->first.str, 5, chassis)) 584*73df0db0SJames Feist { 585*73df0db0SJames Feist return &(*it); 586*73df0db0SJames Feist } 587*73df0db0SJames Feist 588*73df0db0SJames Feist return nullptr; 589b6baeaa4SJames Feist } 590b6baeaa4SJames Feist 59183ff9ab6SJames Feist static CreatePIDRet createPidInterface( 59283ff9ab6SJames Feist const std::shared_ptr<AsyncResp>& response, const std::string& type, 593b6baeaa4SJames Feist nlohmann::json::iterator it, const std::string& path, 59483ff9ab6SJames Feist const dbus::utility::ManagedObjectType& managedObj, bool createNewObject, 59583ff9ab6SJames Feist boost::container::flat_map<std::string, dbus::utility::DbusVariantType>& 59683ff9ab6SJames Feist output, 597*73df0db0SJames Feist std::string& chassis, const std::string& profile) 59883ff9ab6SJames Feist { 59983ff9ab6SJames Feist 6005f2caaefSJames Feist // common deleter 601b6baeaa4SJames Feist if (it.value() == nullptr) 6025f2caaefSJames Feist { 6035f2caaefSJames Feist std::string iface; 6045f2caaefSJames Feist if (type == "PidControllers" || type == "FanControllers") 6055f2caaefSJames Feist { 6065f2caaefSJames Feist iface = pidConfigurationIface; 6075f2caaefSJames Feist } 6085f2caaefSJames Feist else if (type == "FanZones") 6095f2caaefSJames Feist { 6105f2caaefSJames Feist iface = pidZoneConfigurationIface; 6115f2caaefSJames Feist } 6125f2caaefSJames Feist else if (type == "StepwiseControllers") 6135f2caaefSJames Feist { 6145f2caaefSJames Feist iface = stepwiseConfigurationIface; 6155f2caaefSJames Feist } 6165f2caaefSJames Feist else 6175f2caaefSJames Feist { 6185f2caaefSJames Feist BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Type " 6195f2caaefSJames Feist << type; 6205f2caaefSJames Feist messages::propertyUnknown(response->res, type); 6215f2caaefSJames Feist return CreatePIDRet::fail; 6225f2caaefSJames Feist } 6235f2caaefSJames Feist // delete interface 6245f2caaefSJames Feist crow::connections::systemBus->async_method_call( 6255f2caaefSJames Feist [response, path](const boost::system::error_code ec) { 6265f2caaefSJames Feist if (ec) 6275f2caaefSJames Feist { 6285f2caaefSJames Feist BMCWEB_LOG_ERROR << "Error patching " << path << ": " << ec; 6295f2caaefSJames Feist messages::internalError(response->res); 630b6baeaa4SJames Feist return; 6315f2caaefSJames Feist } 632b6baeaa4SJames Feist messages::success(response->res); 6335f2caaefSJames Feist }, 6345f2caaefSJames Feist "xyz.openbmc_project.EntityManager", path, iface, "Delete"); 6355f2caaefSJames Feist return CreatePIDRet::del; 6365f2caaefSJames Feist } 6375f2caaefSJames Feist 638*73df0db0SJames Feist const dbus::utility::ManagedItem* managedItem = nullptr; 639b6baeaa4SJames Feist if (!createNewObject) 640b6baeaa4SJames Feist { 641b6baeaa4SJames Feist // if we aren't creating a new object, we should be able to find it on 642b6baeaa4SJames Feist // d-bus 643*73df0db0SJames Feist managedItem = findChassis(managedObj, it.key(), chassis); 644*73df0db0SJames Feist if (managedItem == nullptr) 645b6baeaa4SJames Feist { 646b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 647b6baeaa4SJames Feist messages::invalidObject(response->res, it.key()); 648b6baeaa4SJames Feist return CreatePIDRet::fail; 649b6baeaa4SJames Feist } 650b6baeaa4SJames Feist } 651b6baeaa4SJames Feist 652*73df0db0SJames Feist if (profile.size() && 653*73df0db0SJames Feist (type == "PidControllers" || type == "FanControllers" || 654*73df0db0SJames Feist type == "StepwiseControllers")) 655*73df0db0SJames Feist { 656*73df0db0SJames Feist if (managedItem == nullptr) 657*73df0db0SJames Feist { 658*73df0db0SJames Feist output["Profiles"] = std::vector<std::string>{profile}; 659*73df0db0SJames Feist } 660*73df0db0SJames Feist else 661*73df0db0SJames Feist { 662*73df0db0SJames Feist std::string interface; 663*73df0db0SJames Feist if (type == "StepwiseControllers") 664*73df0db0SJames Feist { 665*73df0db0SJames Feist interface = stepwiseConfigurationIface; 666*73df0db0SJames Feist } 667*73df0db0SJames Feist else 668*73df0db0SJames Feist { 669*73df0db0SJames Feist interface = pidConfigurationIface; 670*73df0db0SJames Feist } 671*73df0db0SJames Feist auto findConfig = managedItem->second.find(interface); 672*73df0db0SJames Feist if (findConfig == managedItem->second.end()) 673*73df0db0SJames Feist { 674*73df0db0SJames Feist BMCWEB_LOG_ERROR 675*73df0db0SJames Feist << "Failed to find interface in managed object"; 676*73df0db0SJames Feist messages::internalError(response->res); 677*73df0db0SJames Feist return CreatePIDRet::fail; 678*73df0db0SJames Feist } 679*73df0db0SJames Feist auto findProfiles = findConfig->second.find("Profiles"); 680*73df0db0SJames Feist if (findProfiles != findConfig->second.end()) 681*73df0db0SJames Feist { 682*73df0db0SJames Feist const std::vector<std::string>* curProfiles = 683*73df0db0SJames Feist std::get_if<std::vector<std::string>>( 684*73df0db0SJames Feist &(findProfiles->second)); 685*73df0db0SJames Feist if (curProfiles == nullptr) 686*73df0db0SJames Feist { 687*73df0db0SJames Feist BMCWEB_LOG_ERROR << "Illegal profiles in managed object"; 688*73df0db0SJames Feist messages::internalError(response->res); 689*73df0db0SJames Feist return CreatePIDRet::fail; 690*73df0db0SJames Feist } 691*73df0db0SJames Feist if (std::find(curProfiles->begin(), curProfiles->end(), 692*73df0db0SJames Feist profile) == curProfiles->end()) 693*73df0db0SJames Feist { 694*73df0db0SJames Feist std::vector<std::string> newProfiles = *curProfiles; 695*73df0db0SJames Feist newProfiles.push_back(profile); 696*73df0db0SJames Feist output["Profiles"] = newProfiles; 697*73df0db0SJames Feist } 698*73df0db0SJames Feist } 699*73df0db0SJames Feist } 700*73df0db0SJames Feist } 701*73df0db0SJames Feist 70283ff9ab6SJames Feist if (type == "PidControllers" || type == "FanControllers") 70383ff9ab6SJames Feist { 70483ff9ab6SJames Feist if (createNewObject) 70583ff9ab6SJames Feist { 70683ff9ab6SJames Feist output["Class"] = type == "PidControllers" ? std::string("temp") 70783ff9ab6SJames Feist : std::string("fan"); 70883ff9ab6SJames Feist output["Type"] = std::string("Pid"); 70983ff9ab6SJames Feist } 7105f2caaefSJames Feist 7115f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> zones; 7125f2caaefSJames Feist std::optional<std::vector<std::string>> inputs; 7135f2caaefSJames Feist std::optional<std::vector<std::string>> outputs; 7145f2caaefSJames Feist std::map<std::string, std::optional<double>> doubles; 7155f2caaefSJames Feist if (!redfish::json_util::readJson( 716b6baeaa4SJames Feist it.value(), response->res, "Inputs", inputs, "Outputs", outputs, 7175f2caaefSJames Feist "Zones", zones, "FFGainCoefficient", 7185f2caaefSJames Feist doubles["FFGainCoefficient"], "FFOffCoefficient", 7195f2caaefSJames Feist doubles["FFOffCoefficient"], "ICoefficient", 7205f2caaefSJames Feist doubles["ICoefficient"], "ILimitMax", doubles["ILimitMax"], 7215f2caaefSJames Feist "ILimitMin", doubles["ILimitMin"], "OutLimitMax", 7225f2caaefSJames Feist doubles["OutLimitMax"], "OutLimitMin", doubles["OutLimitMin"], 7235f2caaefSJames Feist "PCoefficient", doubles["PCoefficient"], "SetPoint", 7245f2caaefSJames Feist doubles["SetPoint"], "SlewNeg", doubles["SlewNeg"], "SlewPos", 725aad1a257SJames Feist doubles["SlewPos"], "PositiveHysteresis", 726aad1a257SJames Feist doubles["PositiveHysteresis"], "NegativeHysteresis", 727aad1a257SJames Feist doubles["NegativeHysteresis"])) 72883ff9ab6SJames Feist { 7295f2caaefSJames Feist BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property " 730b6baeaa4SJames Feist << it.value().dump(); 7315f2caaefSJames Feist return CreatePIDRet::fail; 73283ff9ab6SJames Feist } 7335f2caaefSJames Feist if (zones) 7345f2caaefSJames Feist { 7355f2caaefSJames Feist std::vector<std::string> zonesStr; 7365f2caaefSJames Feist if (!getZonesFromJsonReq(response, *zones, zonesStr)) 7375f2caaefSJames Feist { 7385f2caaefSJames Feist BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Zones"; 7395f2caaefSJames Feist return CreatePIDRet::fail; 7405f2caaefSJames Feist } 741b6baeaa4SJames Feist if (chassis.empty() && 742b6baeaa4SJames Feist !findChassis(managedObj, zonesStr[0], chassis)) 743b6baeaa4SJames Feist { 744b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 745b6baeaa4SJames Feist messages::invalidObject(response->res, it.key()); 746b6baeaa4SJames Feist return CreatePIDRet::fail; 747b6baeaa4SJames Feist } 748b6baeaa4SJames Feist 7495f2caaefSJames Feist output["Zones"] = std::move(zonesStr); 7505f2caaefSJames Feist } 7515f2caaefSJames Feist if (inputs || outputs) 7525f2caaefSJames Feist { 7535f2caaefSJames Feist std::array<std::optional<std::vector<std::string>>*, 2> containers = 7545f2caaefSJames Feist {&inputs, &outputs}; 7555f2caaefSJames Feist size_t index = 0; 7565f2caaefSJames Feist for (const auto& containerPtr : containers) 7575f2caaefSJames Feist { 7585f2caaefSJames Feist std::optional<std::vector<std::string>>& container = 7595f2caaefSJames Feist *containerPtr; 7605f2caaefSJames Feist if (!container) 7615f2caaefSJames Feist { 7625f2caaefSJames Feist index++; 7635f2caaefSJames Feist continue; 76483ff9ab6SJames Feist } 76583ff9ab6SJames Feist 7665f2caaefSJames Feist for (std::string& value : *container) 76783ff9ab6SJames Feist { 7685f2caaefSJames Feist boost::replace_all(value, "_", " "); 76983ff9ab6SJames Feist } 7705f2caaefSJames Feist std::string key; 7715f2caaefSJames Feist if (index == 0) 7725f2caaefSJames Feist { 7735f2caaefSJames Feist key = "Inputs"; 7745f2caaefSJames Feist } 7755f2caaefSJames Feist else 7765f2caaefSJames Feist { 7775f2caaefSJames Feist key = "Outputs"; 7785f2caaefSJames Feist } 7795f2caaefSJames Feist output[key] = *container; 7805f2caaefSJames Feist index++; 7815f2caaefSJames Feist } 78283ff9ab6SJames Feist } 78383ff9ab6SJames Feist 78483ff9ab6SJames Feist // doubles 7855f2caaefSJames Feist for (const auto& pairs : doubles) 78683ff9ab6SJames Feist { 7875f2caaefSJames Feist if (!pairs.second) 78883ff9ab6SJames Feist { 7895f2caaefSJames Feist continue; 79083ff9ab6SJames Feist } 7915f2caaefSJames Feist BMCWEB_LOG_DEBUG << pairs.first << " = " << *pairs.second; 7925f2caaefSJames Feist output[pairs.first] = *(pairs.second); 7935f2caaefSJames Feist } 79483ff9ab6SJames Feist } 79583ff9ab6SJames Feist 79683ff9ab6SJames Feist else if (type == "FanZones") 79783ff9ab6SJames Feist { 79883ff9ab6SJames Feist output["Type"] = std::string("Pid.Zone"); 79983ff9ab6SJames Feist 8005f2caaefSJames Feist std::optional<nlohmann::json> chassisContainer; 8015f2caaefSJames Feist std::optional<double> failSafePercent; 802d3ec07f8SJames Feist std::optional<double> minThermalOutput; 803b6baeaa4SJames Feist if (!redfish::json_util::readJson(it.value(), response->res, "Chassis", 8045f2caaefSJames Feist chassisContainer, "FailSafePercent", 805d3ec07f8SJames Feist failSafePercent, "MinThermalOutput", 806d3ec07f8SJames Feist minThermalOutput)) 80783ff9ab6SJames Feist { 8085f2caaefSJames Feist BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property " 809b6baeaa4SJames Feist << it.value().dump(); 81083ff9ab6SJames Feist return CreatePIDRet::fail; 81183ff9ab6SJames Feist } 8125f2caaefSJames Feist 8135f2caaefSJames Feist if (chassisContainer) 81483ff9ab6SJames Feist { 8155f2caaefSJames Feist 8165f2caaefSJames Feist std::string chassisId; 8175f2caaefSJames Feist if (!redfish::json_util::readJson(*chassisContainer, response->res, 8185f2caaefSJames Feist "@odata.id", chassisId)) 8195f2caaefSJames Feist { 8205f2caaefSJames Feist BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property " 8215f2caaefSJames Feist << chassisContainer->dump(); 82283ff9ab6SJames Feist return CreatePIDRet::fail; 82383ff9ab6SJames Feist } 82483ff9ab6SJames Feist 82583ff9ab6SJames Feist // /refish/v1/chassis/chassis_name/ 8265f2caaefSJames Feist if (!dbus::utility::getNthStringFromPath(chassisId, 3, chassis)) 82783ff9ab6SJames Feist { 8285f2caaefSJames Feist BMCWEB_LOG_ERROR << "Got invalid path " << chassisId; 8295f2caaefSJames Feist messages::invalidObject(response->res, chassisId); 83083ff9ab6SJames Feist return CreatePIDRet::fail; 83183ff9ab6SJames Feist } 83283ff9ab6SJames Feist } 833d3ec07f8SJames Feist if (minThermalOutput) 83483ff9ab6SJames Feist { 835d3ec07f8SJames Feist output["MinThermalOutput"] = *minThermalOutput; 8365f2caaefSJames Feist } 8375f2caaefSJames Feist if (failSafePercent) 83883ff9ab6SJames Feist { 8395f2caaefSJames Feist output["FailSafePercent"] = *failSafePercent; 8405f2caaefSJames Feist } 8415f2caaefSJames Feist } 8425f2caaefSJames Feist else if (type == "StepwiseControllers") 8435f2caaefSJames Feist { 8445f2caaefSJames Feist output["Type"] = std::string("Stepwise"); 8455f2caaefSJames Feist 8465f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> zones; 8475f2caaefSJames Feist std::optional<std::vector<nlohmann::json>> steps; 8485f2caaefSJames Feist std::optional<std::vector<std::string>> inputs; 8495f2caaefSJames Feist std::optional<double> positiveHysteresis; 8505f2caaefSJames Feist std::optional<double> negativeHysteresis; 851c33a90ecSJames Feist std::optional<std::string> direction; // upper clipping curve vs lower 8525f2caaefSJames Feist if (!redfish::json_util::readJson( 853b6baeaa4SJames Feist it.value(), response->res, "Zones", zones, "Steps", steps, 854b6baeaa4SJames Feist "Inputs", inputs, "PositiveHysteresis", positiveHysteresis, 855c33a90ecSJames Feist "NegativeHysteresis", negativeHysteresis, "Direction", 856c33a90ecSJames Feist direction)) 8575f2caaefSJames Feist { 8585f2caaefSJames Feist BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property " 859b6baeaa4SJames Feist << it.value().dump(); 86083ff9ab6SJames Feist return CreatePIDRet::fail; 86183ff9ab6SJames Feist } 8625f2caaefSJames Feist 8635f2caaefSJames Feist if (zones) 86483ff9ab6SJames Feist { 865b6baeaa4SJames Feist std::vector<std::string> zonesStrs; 866b6baeaa4SJames Feist if (!getZonesFromJsonReq(response, *zones, zonesStrs)) 8675f2caaefSJames Feist { 8685f2caaefSJames Feist BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Zones"; 86983ff9ab6SJames Feist return CreatePIDRet::fail; 87083ff9ab6SJames Feist } 871b6baeaa4SJames Feist if (chassis.empty() && 872b6baeaa4SJames Feist !findChassis(managedObj, zonesStrs[0], chassis)) 873b6baeaa4SJames Feist { 874b6baeaa4SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config patch"; 875b6baeaa4SJames Feist messages::invalidObject(response->res, it.key()); 876b6baeaa4SJames Feist return CreatePIDRet::fail; 877b6baeaa4SJames Feist } 878b6baeaa4SJames Feist output["Zones"] = std::move(zonesStrs); 8795f2caaefSJames Feist } 8805f2caaefSJames Feist if (steps) 8815f2caaefSJames Feist { 8825f2caaefSJames Feist std::vector<double> readings; 8835f2caaefSJames Feist std::vector<double> outputs; 8845f2caaefSJames Feist for (auto& step : *steps) 8855f2caaefSJames Feist { 8865f2caaefSJames Feist double target; 887b01bf299SEd Tanous double output; 8885f2caaefSJames Feist 8895f2caaefSJames Feist if (!redfish::json_util::readJson(step, response->res, "Target", 890b01bf299SEd Tanous target, "Output", output)) 8915f2caaefSJames Feist { 8925f2caaefSJames Feist BMCWEB_LOG_ERROR << "Line:" << __LINE__ 893b6baeaa4SJames Feist << ", Illegal Property " 894b6baeaa4SJames Feist << it.value().dump(); 8955f2caaefSJames Feist return CreatePIDRet::fail; 8965f2caaefSJames Feist } 8975f2caaefSJames Feist readings.emplace_back(target); 898b01bf299SEd Tanous outputs.emplace_back(output); 8995f2caaefSJames Feist } 9005f2caaefSJames Feist output["Reading"] = std::move(readings); 9015f2caaefSJames Feist output["Output"] = std::move(outputs); 9025f2caaefSJames Feist } 9035f2caaefSJames Feist if (inputs) 9045f2caaefSJames Feist { 9055f2caaefSJames Feist for (std::string& value : *inputs) 9065f2caaefSJames Feist { 9075f2caaefSJames Feist boost::replace_all(value, "_", " "); 9085f2caaefSJames Feist } 9095f2caaefSJames Feist output["Inputs"] = std::move(*inputs); 9105f2caaefSJames Feist } 9115f2caaefSJames Feist if (negativeHysteresis) 9125f2caaefSJames Feist { 9135f2caaefSJames Feist output["NegativeHysteresis"] = *negativeHysteresis; 9145f2caaefSJames Feist } 9155f2caaefSJames Feist if (positiveHysteresis) 9165f2caaefSJames Feist { 9175f2caaefSJames Feist output["PositiveHysteresis"] = *positiveHysteresis; 91883ff9ab6SJames Feist } 919c33a90ecSJames Feist if (direction) 920c33a90ecSJames Feist { 921c33a90ecSJames Feist constexpr const std::array<const char*, 2> allowedDirections = { 922c33a90ecSJames Feist "Ceiling", "Floor"}; 923c33a90ecSJames Feist if (std::find(allowedDirections.begin(), allowedDirections.end(), 924c33a90ecSJames Feist *direction) == allowedDirections.end()) 925c33a90ecSJames Feist { 926c33a90ecSJames Feist messages::propertyValueTypeError(response->res, "Direction", 927c33a90ecSJames Feist *direction); 928c33a90ecSJames Feist return CreatePIDRet::fail; 929c33a90ecSJames Feist } 930c33a90ecSJames Feist output["Class"] = *direction; 931c33a90ecSJames Feist } 93283ff9ab6SJames Feist } 93383ff9ab6SJames Feist else 93483ff9ab6SJames Feist { 9355f2caaefSJames Feist BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Type " << type; 93635a62c7cSJason M. Bills messages::propertyUnknown(response->res, type); 93783ff9ab6SJames Feist return CreatePIDRet::fail; 93883ff9ab6SJames Feist } 93983ff9ab6SJames Feist return CreatePIDRet::patch; 94083ff9ab6SJames Feist } 941*73df0db0SJames Feist struct GetPIDValues : std::enable_shared_from_this<GetPIDValues> 942*73df0db0SJames Feist { 94383ff9ab6SJames Feist 944*73df0db0SJames Feist GetPIDValues(const std::shared_ptr<AsyncResp>& asyncResp) : 945*73df0db0SJames Feist asyncResp(asyncResp) 946*73df0db0SJames Feist 9471abe55efSEd Tanous { 9489c310685SBorawski.Lukasz } 9499c310685SBorawski.Lukasz 950*73df0db0SJames Feist void run() 9515b4aa86bSJames Feist { 952*73df0db0SJames Feist std::shared_ptr<GetPIDValues> self = shared_from_this(); 953*73df0db0SJames Feist 954*73df0db0SJames Feist // get all configurations 9555b4aa86bSJames Feist crow::connections::systemBus->async_method_call( 956*73df0db0SJames Feist [self](const boost::system::error_code ec, 9575b4aa86bSJames Feist const crow::openbmc_mapper::GetSubTreeType& subtree) { 9585b4aa86bSJames Feist if (ec) 9595b4aa86bSJames Feist { 9605b4aa86bSJames Feist BMCWEB_LOG_ERROR << ec; 961*73df0db0SJames Feist messages::internalError(self->asyncResp->res); 962*73df0db0SJames Feist return; 963*73df0db0SJames Feist } 964*73df0db0SJames Feist self->subtree = subtree; 965*73df0db0SJames Feist }, 966*73df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", 967*73df0db0SJames Feist "/xyz/openbmc_project/object_mapper", 968*73df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 969*73df0db0SJames Feist std::array<const char*, 4>{ 970*73df0db0SJames Feist pidConfigurationIface, pidZoneConfigurationIface, 971*73df0db0SJames Feist objectManagerIface, stepwiseConfigurationIface}); 972*73df0db0SJames Feist 973*73df0db0SJames Feist // at the same time get the selected profile 974*73df0db0SJames Feist crow::connections::systemBus->async_method_call( 975*73df0db0SJames Feist [self](const boost::system::error_code ec, 976*73df0db0SJames Feist const crow::openbmc_mapper::GetSubTreeType& subtree) { 977*73df0db0SJames Feist if (ec || subtree.empty()) 978*73df0db0SJames Feist { 979*73df0db0SJames Feist return; 980*73df0db0SJames Feist } 981*73df0db0SJames Feist if (subtree[0].second.size() != 1) 982*73df0db0SJames Feist { 983*73df0db0SJames Feist // invalid mapper response, should never happen 984*73df0db0SJames Feist BMCWEB_LOG_ERROR << "GetPIDValues: Mapper Error"; 985*73df0db0SJames Feist messages::internalError(self->asyncResp->res); 9865b4aa86bSJames Feist return; 9875b4aa86bSJames Feist } 9885b4aa86bSJames Feist 989*73df0db0SJames Feist const std::string& path = subtree[0].first; 990*73df0db0SJames Feist const std::string& owner = subtree[0].second[0].first; 991*73df0db0SJames Feist crow::connections::systemBus->async_method_call( 992*73df0db0SJames Feist [path, owner, self]( 993*73df0db0SJames Feist const boost::system::error_code ec, 994*73df0db0SJames Feist const boost::container::flat_map< 995*73df0db0SJames Feist std::string, std::variant<std::vector<std::string>, 996*73df0db0SJames Feist std::string>>& resp) { 997*73df0db0SJames Feist if (ec) 998*73df0db0SJames Feist { 999*73df0db0SJames Feist BMCWEB_LOG_ERROR << "GetPIDValues: Can't get " 1000*73df0db0SJames Feist "thermalModeIface " 1001*73df0db0SJames Feist << path; 1002*73df0db0SJames Feist messages::internalError(self->asyncResp->res); 1003*73df0db0SJames Feist return; 1004*73df0db0SJames Feist } 1005*73df0db0SJames Feist const std::string* current; 1006*73df0db0SJames Feist const std::vector<std::string>* supported; 1007*73df0db0SJames Feist for (auto& [key, value] : resp) 1008*73df0db0SJames Feist { 1009*73df0db0SJames Feist if (key == "Current") 1010*73df0db0SJames Feist { 1011*73df0db0SJames Feist current = std::get_if<std::string>(&value); 1012*73df0db0SJames Feist if (current == nullptr) 1013*73df0db0SJames Feist { 1014*73df0db0SJames Feist BMCWEB_LOG_ERROR 1015*73df0db0SJames Feist << "GetPIDValues: thermal mode " 1016*73df0db0SJames Feist "iface invalid " 1017*73df0db0SJames Feist << path; 1018*73df0db0SJames Feist messages::internalError( 1019*73df0db0SJames Feist self->asyncResp->res); 1020*73df0db0SJames Feist return; 1021*73df0db0SJames Feist } 1022*73df0db0SJames Feist } 1023*73df0db0SJames Feist if (key == "Supported") 1024*73df0db0SJames Feist { 1025*73df0db0SJames Feist supported = 1026*73df0db0SJames Feist std::get_if<std::vector<std::string>>( 1027*73df0db0SJames Feist &value); 1028*73df0db0SJames Feist if (supported == nullptr) 1029*73df0db0SJames Feist { 1030*73df0db0SJames Feist BMCWEB_LOG_ERROR 1031*73df0db0SJames Feist << "GetPIDValues: thermal mode " 1032*73df0db0SJames Feist "iface invalid" 1033*73df0db0SJames Feist << path; 1034*73df0db0SJames Feist messages::internalError( 1035*73df0db0SJames Feist self->asyncResp->res); 1036*73df0db0SJames Feist return; 1037*73df0db0SJames Feist } 1038*73df0db0SJames Feist } 1039*73df0db0SJames Feist } 1040*73df0db0SJames Feist if (current == nullptr || supported == nullptr) 1041*73df0db0SJames Feist { 1042*73df0db0SJames Feist BMCWEB_LOG_ERROR << "GetPIDValues: thermal mode " 1043*73df0db0SJames Feist "iface invalid " 1044*73df0db0SJames Feist << path; 1045*73df0db0SJames Feist messages::internalError(self->asyncResp->res); 1046*73df0db0SJames Feist return; 1047*73df0db0SJames Feist } 1048*73df0db0SJames Feist self->currentProfile = *current; 1049*73df0db0SJames Feist self->supportedProfiles = *supported; 1050*73df0db0SJames Feist }, 1051*73df0db0SJames Feist owner, path, "org.freedesktop.DBus.Properties", "GetAll", 1052*73df0db0SJames Feist thermalModeIface); 1053*73df0db0SJames Feist }, 1054*73df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", 1055*73df0db0SJames Feist "/xyz/openbmc_project/object_mapper", 1056*73df0db0SJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 1057*73df0db0SJames Feist std::array<const char*, 1>{thermalModeIface}); 1058*73df0db0SJames Feist } 1059*73df0db0SJames Feist 1060*73df0db0SJames Feist ~GetPIDValues() 1061*73df0db0SJames Feist { 1062*73df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 1063*73df0db0SJames Feist { 1064*73df0db0SJames Feist return; 1065*73df0db0SJames Feist } 10665b4aa86bSJames Feist // create map of <connection, path to objMgr>> 1067*73df0db0SJames Feist boost::container::flat_map<std::string, std::string> objectMgrPaths; 10686bce33bcSJames Feist boost::container::flat_set<std::string> calledConnections; 10695b4aa86bSJames Feist for (const auto& pathGroup : subtree) 10705b4aa86bSJames Feist { 10715b4aa86bSJames Feist for (const auto& connectionGroup : pathGroup.second) 10725b4aa86bSJames Feist { 10736bce33bcSJames Feist auto findConnection = 10746bce33bcSJames Feist calledConnections.find(connectionGroup.first); 10756bce33bcSJames Feist if (findConnection != calledConnections.end()) 10766bce33bcSJames Feist { 10776bce33bcSJames Feist break; 10786bce33bcSJames Feist } 1079*73df0db0SJames Feist for (const std::string& interface : connectionGroup.second) 10805b4aa86bSJames Feist { 10815b4aa86bSJames Feist if (interface == objectManagerIface) 10825b4aa86bSJames Feist { 1083*73df0db0SJames Feist objectMgrPaths[connectionGroup.first] = pathGroup.first; 10845b4aa86bSJames Feist } 10855b4aa86bSJames Feist // this list is alphabetical, so we 10865b4aa86bSJames Feist // should have found the objMgr by now 10875b4aa86bSJames Feist if (interface == pidConfigurationIface || 1088b7a08d04SJames Feist interface == pidZoneConfigurationIface || 1089b7a08d04SJames Feist interface == stepwiseConfigurationIface) 10905b4aa86bSJames Feist { 10915b4aa86bSJames Feist auto findObjMgr = 10925b4aa86bSJames Feist objectMgrPaths.find(connectionGroup.first); 10935b4aa86bSJames Feist if (findObjMgr == objectMgrPaths.end()) 10945b4aa86bSJames Feist { 10955b4aa86bSJames Feist BMCWEB_LOG_DEBUG << connectionGroup.first 10965b4aa86bSJames Feist << "Has no Object Manager"; 10975b4aa86bSJames Feist continue; 10985b4aa86bSJames Feist } 10996bce33bcSJames Feist 11006bce33bcSJames Feist calledConnections.insert(connectionGroup.first); 11016bce33bcSJames Feist 1102*73df0db0SJames Feist asyncPopulatePid(findObjMgr->first, findObjMgr->second, 1103*73df0db0SJames Feist currentProfile, supportedProfiles, 1104*73df0db0SJames Feist asyncResp); 11055b4aa86bSJames Feist break; 11065b4aa86bSJames Feist } 11075b4aa86bSJames Feist } 11085b4aa86bSJames Feist } 11095b4aa86bSJames Feist } 1110*73df0db0SJames Feist } 1111*73df0db0SJames Feist 1112*73df0db0SJames Feist std::vector<std::string> supportedProfiles; 1113*73df0db0SJames Feist std::string currentProfile; 1114*73df0db0SJames Feist crow::openbmc_mapper::GetSubTreeType subtree; 1115*73df0db0SJames Feist std::shared_ptr<AsyncResp> asyncResp; 1116*73df0db0SJames Feist }; 1117*73df0db0SJames Feist 1118*73df0db0SJames Feist struct SetPIDValues : std::enable_shared_from_this<SetPIDValues> 1119*73df0db0SJames Feist { 1120*73df0db0SJames Feist 1121*73df0db0SJames Feist SetPIDValues(const std::shared_ptr<AsyncResp>& asyncResp, 1122*73df0db0SJames Feist nlohmann::json& data) : 1123*73df0db0SJames Feist asyncResp(asyncResp) 1124*73df0db0SJames Feist { 1125*73df0db0SJames Feist 1126*73df0db0SJames Feist std::optional<nlohmann::json> pidControllers; 1127*73df0db0SJames Feist std::optional<nlohmann::json> fanControllers; 1128*73df0db0SJames Feist std::optional<nlohmann::json> fanZones; 1129*73df0db0SJames Feist std::optional<nlohmann::json> stepwiseControllers; 1130*73df0db0SJames Feist 1131*73df0db0SJames Feist if (!redfish::json_util::readJson( 1132*73df0db0SJames Feist data, asyncResp->res, "PidControllers", pidControllers, 1133*73df0db0SJames Feist "FanControllers", fanControllers, "FanZones", fanZones, 1134*73df0db0SJames Feist "StepwiseControllers", stepwiseControllers, "Profile", profile)) 1135*73df0db0SJames Feist { 1136*73df0db0SJames Feist BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property " 1137*73df0db0SJames Feist << data.dump(); 1138*73df0db0SJames Feist return; 1139*73df0db0SJames Feist } 1140*73df0db0SJames Feist configuration.emplace_back("PidControllers", std::move(pidControllers)); 1141*73df0db0SJames Feist configuration.emplace_back("FanControllers", std::move(fanControllers)); 1142*73df0db0SJames Feist configuration.emplace_back("FanZones", std::move(fanZones)); 1143*73df0db0SJames Feist configuration.emplace_back("StepwiseControllers", 1144*73df0db0SJames Feist std::move(stepwiseControllers)); 1145*73df0db0SJames Feist } 1146*73df0db0SJames Feist void run() 1147*73df0db0SJames Feist { 1148*73df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 1149*73df0db0SJames Feist { 1150*73df0db0SJames Feist return; 1151*73df0db0SJames Feist } 1152*73df0db0SJames Feist 1153*73df0db0SJames Feist std::shared_ptr<SetPIDValues> self = shared_from_this(); 1154*73df0db0SJames Feist 1155*73df0db0SJames Feist // todo(james): might make sense to do a mapper call here if this 1156*73df0db0SJames Feist // interface gets more traction 1157*73df0db0SJames Feist crow::connections::systemBus->async_method_call( 1158*73df0db0SJames Feist [self](const boost::system::error_code ec, 1159*73df0db0SJames Feist dbus::utility::ManagedObjectType& managedObj) { 1160*73df0db0SJames Feist if (ec) 1161*73df0db0SJames Feist { 1162*73df0db0SJames Feist BMCWEB_LOG_ERROR << "Error communicating to Entity Manager"; 1163*73df0db0SJames Feist messages::internalError(self->asyncResp->res); 1164*73df0db0SJames Feist return; 1165*73df0db0SJames Feist } 1166*73df0db0SJames Feist self->managedObj = std::move(managedObj); 1167*73df0db0SJames Feist }, 1168*73df0db0SJames Feist "xyz.openbmc_project.EntityManager", "/", objectManagerIface, 1169*73df0db0SJames Feist "GetManagedObjects"); 1170*73df0db0SJames Feist 1171*73df0db0SJames Feist // at the same time get the profile information 1172*73df0db0SJames Feist crow::connections::systemBus->async_method_call( 1173*73df0db0SJames Feist [self](const boost::system::error_code ec, 1174*73df0db0SJames Feist const crow::openbmc_mapper::GetSubTreeType& subtree) { 1175*73df0db0SJames Feist if (ec || subtree.empty()) 1176*73df0db0SJames Feist { 1177*73df0db0SJames Feist return; 1178*73df0db0SJames Feist } 1179*73df0db0SJames Feist if (subtree[0].second.empty()) 1180*73df0db0SJames Feist { 1181*73df0db0SJames Feist // invalid mapper response, should never happen 1182*73df0db0SJames Feist BMCWEB_LOG_ERROR << "SetPIDValues: Mapper Error"; 1183*73df0db0SJames Feist messages::internalError(self->asyncResp->res); 1184*73df0db0SJames Feist return; 1185*73df0db0SJames Feist } 1186*73df0db0SJames Feist 1187*73df0db0SJames Feist const std::string& path = subtree[0].first; 1188*73df0db0SJames Feist const std::string& owner = subtree[0].second[0].first; 1189*73df0db0SJames Feist crow::connections::systemBus->async_method_call( 1190*73df0db0SJames Feist [self, path, owner]( 1191*73df0db0SJames Feist const boost::system::error_code ec, 1192*73df0db0SJames Feist const boost::container::flat_map< 1193*73df0db0SJames Feist std::string, std::variant<std::vector<std::string>, 1194*73df0db0SJames Feist std::string>>& resp) { 1195*73df0db0SJames Feist if (ec) 1196*73df0db0SJames Feist { 1197*73df0db0SJames Feist BMCWEB_LOG_ERROR << "SetPIDValues: Can't get " 1198*73df0db0SJames Feist "thermalModeIface " 1199*73df0db0SJames Feist << path; 1200*73df0db0SJames Feist messages::internalError(self->asyncResp->res); 1201*73df0db0SJames Feist return; 1202*73df0db0SJames Feist } 1203*73df0db0SJames Feist const std::string* current; 1204*73df0db0SJames Feist const std::vector<std::string>* supported; 1205*73df0db0SJames Feist for (auto& [key, value] : resp) 1206*73df0db0SJames Feist { 1207*73df0db0SJames Feist if (key == "Current") 1208*73df0db0SJames Feist { 1209*73df0db0SJames Feist current = std::get_if<std::string>(&value); 1210*73df0db0SJames Feist if (current == nullptr) 1211*73df0db0SJames Feist { 1212*73df0db0SJames Feist BMCWEB_LOG_ERROR 1213*73df0db0SJames Feist << "SetPIDValues: thermal mode " 1214*73df0db0SJames Feist "iface invalid " 1215*73df0db0SJames Feist << path; 1216*73df0db0SJames Feist messages::internalError( 1217*73df0db0SJames Feist self->asyncResp->res); 1218*73df0db0SJames Feist return; 1219*73df0db0SJames Feist } 1220*73df0db0SJames Feist } 1221*73df0db0SJames Feist if (key == "Supported") 1222*73df0db0SJames Feist { 1223*73df0db0SJames Feist supported = 1224*73df0db0SJames Feist std::get_if<std::vector<std::string>>( 1225*73df0db0SJames Feist &value); 1226*73df0db0SJames Feist if (supported == nullptr) 1227*73df0db0SJames Feist { 1228*73df0db0SJames Feist BMCWEB_LOG_ERROR 1229*73df0db0SJames Feist << "SetPIDValues: thermal mode " 1230*73df0db0SJames Feist "iface invalid" 1231*73df0db0SJames Feist << path; 1232*73df0db0SJames Feist messages::internalError( 1233*73df0db0SJames Feist self->asyncResp->res); 1234*73df0db0SJames Feist return; 1235*73df0db0SJames Feist } 1236*73df0db0SJames Feist } 1237*73df0db0SJames Feist } 1238*73df0db0SJames Feist if (current == nullptr || supported == nullptr) 1239*73df0db0SJames Feist { 1240*73df0db0SJames Feist BMCWEB_LOG_ERROR << "SetPIDValues: thermal mode " 1241*73df0db0SJames Feist "iface invalid " 1242*73df0db0SJames Feist << path; 1243*73df0db0SJames Feist messages::internalError(self->asyncResp->res); 1244*73df0db0SJames Feist return; 1245*73df0db0SJames Feist } 1246*73df0db0SJames Feist self->currentProfile = *current; 1247*73df0db0SJames Feist self->supportedProfiles = *supported; 1248*73df0db0SJames Feist self->profileConnection = owner; 1249*73df0db0SJames Feist self->profilePath = path; 1250*73df0db0SJames Feist }, 1251*73df0db0SJames Feist owner, path, "org.freedesktop.DBus.Properties", "GetAll", 1252*73df0db0SJames Feist thermalModeIface); 12535b4aa86bSJames Feist }, 12545b4aa86bSJames Feist "xyz.openbmc_project.ObjectMapper", 12555b4aa86bSJames Feist "/xyz/openbmc_project/object_mapper", 12565b4aa86bSJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 1257*73df0db0SJames Feist std::array<const char*, 1>{thermalModeIface}); 1258*73df0db0SJames Feist } 1259*73df0db0SJames Feist ~SetPIDValues() 1260*73df0db0SJames Feist { 1261*73df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 1262*73df0db0SJames Feist { 1263*73df0db0SJames Feist return; 12645b4aa86bSJames Feist } 12655b4aa86bSJames Feist 1266*73df0db0SJames Feist std::shared_ptr<AsyncResp> response = asyncResp; 1267*73df0db0SJames Feist 1268*73df0db0SJames Feist if (profile) 1269*73df0db0SJames Feist { 1270*73df0db0SJames Feist if (std::find(supportedProfiles.begin(), supportedProfiles.end(), 1271*73df0db0SJames Feist *profile) == supportedProfiles.end()) 1272*73df0db0SJames Feist { 1273*73df0db0SJames Feist messages::actionParameterUnknown(response->res, "Profile", 1274*73df0db0SJames Feist *profile); 1275*73df0db0SJames Feist return; 1276*73df0db0SJames Feist } 1277*73df0db0SJames Feist currentProfile = *profile; 1278*73df0db0SJames Feist crow::connections::systemBus->async_method_call( 1279*73df0db0SJames Feist [response](const boost::system::error_code ec) { 1280*73df0db0SJames Feist if (ec) 1281*73df0db0SJames Feist { 1282*73df0db0SJames Feist BMCWEB_LOG_ERROR << "Error patching profile" << ec; 1283*73df0db0SJames Feist messages::internalError(response->res); 1284*73df0db0SJames Feist } 1285*73df0db0SJames Feist }, 1286*73df0db0SJames Feist profileConnection, profilePath, 1287*73df0db0SJames Feist "org.freedesktop.DBus.Properties", "Set", thermalModeIface, 1288*73df0db0SJames Feist "Current", std::variant<std::string>(*profile)); 1289*73df0db0SJames Feist } 1290*73df0db0SJames Feist 1291*73df0db0SJames Feist for (auto& containerPair : configuration) 1292*73df0db0SJames Feist { 1293*73df0db0SJames Feist auto& container = containerPair.second; 1294*73df0db0SJames Feist if (!container) 1295*73df0db0SJames Feist { 1296*73df0db0SJames Feist continue; 1297*73df0db0SJames Feist } 1298*73df0db0SJames Feist std::string& type = containerPair.first; 1299*73df0db0SJames Feist 1300*73df0db0SJames Feist for (nlohmann::json::iterator it = container->begin(); 1301*73df0db0SJames Feist it != container->end(); it++) 1302*73df0db0SJames Feist { 1303*73df0db0SJames Feist const auto& name = it.key(); 1304*73df0db0SJames Feist auto pathItr = 1305*73df0db0SJames Feist std::find_if(managedObj.begin(), managedObj.end(), 1306*73df0db0SJames Feist [&name](const auto& obj) { 1307*73df0db0SJames Feist return boost::algorithm::ends_with( 1308*73df0db0SJames Feist obj.first.str, "/" + name); 1309*73df0db0SJames Feist }); 1310*73df0db0SJames Feist boost::container::flat_map<std::string, 1311*73df0db0SJames Feist dbus::utility::DbusVariantType> 1312*73df0db0SJames Feist output; 1313*73df0db0SJames Feist 1314*73df0db0SJames Feist output.reserve(16); // The pid interface length 1315*73df0db0SJames Feist 1316*73df0db0SJames Feist // determines if we're patching entity-manager or 1317*73df0db0SJames Feist // creating a new object 1318*73df0db0SJames Feist bool createNewObject = (pathItr == managedObj.end()); 1319*73df0db0SJames Feist std::string iface; 1320*73df0db0SJames Feist if (type == "PidControllers" || type == "FanControllers") 1321*73df0db0SJames Feist { 1322*73df0db0SJames Feist iface = pidConfigurationIface; 1323*73df0db0SJames Feist if (!createNewObject && 1324*73df0db0SJames Feist pathItr->second.find(pidConfigurationIface) == 1325*73df0db0SJames Feist pathItr->second.end()) 1326*73df0db0SJames Feist { 1327*73df0db0SJames Feist createNewObject = true; 1328*73df0db0SJames Feist } 1329*73df0db0SJames Feist } 1330*73df0db0SJames Feist else if (type == "FanZones") 1331*73df0db0SJames Feist { 1332*73df0db0SJames Feist iface = pidZoneConfigurationIface; 1333*73df0db0SJames Feist if (!createNewObject && 1334*73df0db0SJames Feist pathItr->second.find(pidZoneConfigurationIface) == 1335*73df0db0SJames Feist pathItr->second.end()) 1336*73df0db0SJames Feist { 1337*73df0db0SJames Feist 1338*73df0db0SJames Feist createNewObject = true; 1339*73df0db0SJames Feist } 1340*73df0db0SJames Feist } 1341*73df0db0SJames Feist else if (type == "StepwiseControllers") 1342*73df0db0SJames Feist { 1343*73df0db0SJames Feist iface = stepwiseConfigurationIface; 1344*73df0db0SJames Feist if (!createNewObject && 1345*73df0db0SJames Feist pathItr->second.find(stepwiseConfigurationIface) == 1346*73df0db0SJames Feist pathItr->second.end()) 1347*73df0db0SJames Feist { 1348*73df0db0SJames Feist createNewObject = true; 1349*73df0db0SJames Feist } 1350*73df0db0SJames Feist } 1351*73df0db0SJames Feist BMCWEB_LOG_DEBUG << "Create new = " << createNewObject << "\n"; 1352*73df0db0SJames Feist output["Name"] = boost::replace_all_copy(name, "_", " "); 1353*73df0db0SJames Feist 1354*73df0db0SJames Feist std::string chassis; 1355*73df0db0SJames Feist CreatePIDRet ret = createPidInterface( 1356*73df0db0SJames Feist response, type, it, pathItr->first.str, managedObj, 1357*73df0db0SJames Feist createNewObject, output, chassis, currentProfile); 1358*73df0db0SJames Feist if (ret == CreatePIDRet::fail) 1359*73df0db0SJames Feist { 1360*73df0db0SJames Feist return; 1361*73df0db0SJames Feist } 1362*73df0db0SJames Feist else if (ret == CreatePIDRet::del) 1363*73df0db0SJames Feist { 1364*73df0db0SJames Feist continue; 1365*73df0db0SJames Feist } 1366*73df0db0SJames Feist 1367*73df0db0SJames Feist if (!createNewObject) 1368*73df0db0SJames Feist { 1369*73df0db0SJames Feist for (const auto& property : output) 1370*73df0db0SJames Feist { 1371*73df0db0SJames Feist crow::connections::systemBus->async_method_call( 1372*73df0db0SJames Feist [response, 1373*73df0db0SJames Feist propertyName{std::string(property.first)}]( 1374*73df0db0SJames Feist const boost::system::error_code ec) { 1375*73df0db0SJames Feist if (ec) 1376*73df0db0SJames Feist { 1377*73df0db0SJames Feist BMCWEB_LOG_ERROR << "Error patching " 1378*73df0db0SJames Feist << propertyName << ": " 1379*73df0db0SJames Feist << ec; 1380*73df0db0SJames Feist messages::internalError(response->res); 1381*73df0db0SJames Feist return; 1382*73df0db0SJames Feist } 1383*73df0db0SJames Feist messages::success(response->res); 1384*73df0db0SJames Feist }, 1385*73df0db0SJames Feist "xyz.openbmc_project.EntityManager", 1386*73df0db0SJames Feist pathItr->first.str, 1387*73df0db0SJames Feist "org.freedesktop.DBus.Properties", "Set", iface, 1388*73df0db0SJames Feist property.first, property.second); 1389*73df0db0SJames Feist } 1390*73df0db0SJames Feist } 1391*73df0db0SJames Feist else 1392*73df0db0SJames Feist { 1393*73df0db0SJames Feist if (chassis.empty()) 1394*73df0db0SJames Feist { 1395*73df0db0SJames Feist BMCWEB_LOG_ERROR << "Failed to get chassis from config"; 1396*73df0db0SJames Feist messages::invalidObject(response->res, name); 1397*73df0db0SJames Feist return; 1398*73df0db0SJames Feist } 1399*73df0db0SJames Feist 1400*73df0db0SJames Feist bool foundChassis = false; 1401*73df0db0SJames Feist for (const auto& obj : managedObj) 1402*73df0db0SJames Feist { 1403*73df0db0SJames Feist if (boost::algorithm::ends_with(obj.first.str, chassis)) 1404*73df0db0SJames Feist { 1405*73df0db0SJames Feist chassis = obj.first.str; 1406*73df0db0SJames Feist foundChassis = true; 1407*73df0db0SJames Feist break; 1408*73df0db0SJames Feist } 1409*73df0db0SJames Feist } 1410*73df0db0SJames Feist if (!foundChassis) 1411*73df0db0SJames Feist { 1412*73df0db0SJames Feist BMCWEB_LOG_ERROR << "Failed to find chassis on dbus"; 1413*73df0db0SJames Feist messages::resourceMissingAtURI( 1414*73df0db0SJames Feist response->res, "/redfish/v1/Chassis/" + chassis); 1415*73df0db0SJames Feist return; 1416*73df0db0SJames Feist } 1417*73df0db0SJames Feist 1418*73df0db0SJames Feist crow::connections::systemBus->async_method_call( 1419*73df0db0SJames Feist [response](const boost::system::error_code ec) { 1420*73df0db0SJames Feist if (ec) 1421*73df0db0SJames Feist { 1422*73df0db0SJames Feist BMCWEB_LOG_ERROR << "Error Adding Pid Object " 1423*73df0db0SJames Feist << ec; 1424*73df0db0SJames Feist messages::internalError(response->res); 1425*73df0db0SJames Feist return; 1426*73df0db0SJames Feist } 1427*73df0db0SJames Feist messages::success(response->res); 1428*73df0db0SJames Feist }, 1429*73df0db0SJames Feist "xyz.openbmc_project.EntityManager", chassis, 1430*73df0db0SJames Feist "xyz.openbmc_project.AddObject", "AddObject", output); 1431*73df0db0SJames Feist } 1432*73df0db0SJames Feist } 1433*73df0db0SJames Feist } 1434*73df0db0SJames Feist } 1435*73df0db0SJames Feist std::shared_ptr<AsyncResp> asyncResp; 1436*73df0db0SJames Feist std::vector<std::pair<std::string, std::optional<nlohmann::json>>> 1437*73df0db0SJames Feist configuration; 1438*73df0db0SJames Feist std::optional<std::string> profile; 1439*73df0db0SJames Feist dbus::utility::ManagedObjectType managedObj; 1440*73df0db0SJames Feist std::vector<std::string> supportedProfiles; 1441*73df0db0SJames Feist std::string currentProfile; 1442*73df0db0SJames Feist std::string profileConnection; 1443*73df0db0SJames Feist std::string profilePath; 1444*73df0db0SJames Feist }; 1445*73df0db0SJames Feist 1446*73df0db0SJames Feist class Manager : public Node 1447*73df0db0SJames Feist { 1448*73df0db0SJames Feist public: 1449*73df0db0SJames Feist Manager(CrowApp& app) : Node(app, "/redfish/v1/Managers/bmc/") 1450*73df0db0SJames Feist { 1451*73df0db0SJames Feist uuid = app.template getMiddleware<crow::persistent_data::Middleware>() 1452*73df0db0SJames Feist .systemUuid; 1453*73df0db0SJames Feist entityPrivileges = { 1454*73df0db0SJames Feist {boost::beast::http::verb::get, {{"Login"}}}, 1455*73df0db0SJames Feist {boost::beast::http::verb::head, {{"Login"}}}, 1456*73df0db0SJames Feist {boost::beast::http::verb::patch, {{"ConfigureManager"}}}, 1457*73df0db0SJames Feist {boost::beast::http::verb::put, {{"ConfigureManager"}}}, 1458*73df0db0SJames Feist {boost::beast::http::verb::delete_, {{"ConfigureManager"}}}, 1459*73df0db0SJames Feist {boost::beast::http::verb::post, {{"ConfigureManager"}}}}; 1460*73df0db0SJames Feist } 1461*73df0db0SJames Feist 1462*73df0db0SJames Feist private: 146355c7b7a2SEd Tanous void doGet(crow::Response& res, const crow::Request& req, 14641abe55efSEd Tanous const std::vector<std::string>& params) override 14651abe55efSEd Tanous { 14660f74e643SEd Tanous res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc"; 14670f74e643SEd Tanous res.jsonValue["@odata.type"] = "#Manager.v1_3_0.Manager"; 14680f74e643SEd Tanous res.jsonValue["@odata.context"] = 14690f74e643SEd Tanous "/redfish/v1/$metadata#Manager.Manager"; 14700f74e643SEd Tanous res.jsonValue["Id"] = "bmc"; 14710f74e643SEd Tanous res.jsonValue["Name"] = "OpenBmc Manager"; 14720f74e643SEd Tanous res.jsonValue["Description"] = "Baseboard Management Controller"; 14730f74e643SEd Tanous res.jsonValue["PowerState"] = "On"; 1474029573d4SEd Tanous res.jsonValue["Status"] = {{"State", "Enabled"}, {"Health", "OK"}}; 14750f74e643SEd Tanous res.jsonValue["ManagerType"] = "BMC"; 14763602e232SEd Tanous res.jsonValue["UUID"] = systemd_utils::getUuid(); 14773602e232SEd Tanous res.jsonValue["ServiceEntryPointUUID"] = uuid; 14780f74e643SEd Tanous res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model 14790f74e643SEd Tanous 14800f74e643SEd Tanous res.jsonValue["LogServices"] = { 14810f74e643SEd Tanous {"@odata.id", "/redfish/v1/Managers/bmc/LogServices"}}; 14820f74e643SEd Tanous 14830f74e643SEd Tanous res.jsonValue["NetworkProtocol"] = { 14840f74e643SEd Tanous {"@odata.id", "/redfish/v1/Managers/bmc/NetworkProtocol"}}; 14850f74e643SEd Tanous 14860f74e643SEd Tanous res.jsonValue["EthernetInterfaces"] = { 14870f74e643SEd Tanous {"@odata.id", "/redfish/v1/Managers/bmc/EthernetInterfaces"}}; 14880f74e643SEd Tanous // default oem data 14890f74e643SEd Tanous nlohmann::json& oem = res.jsonValue["Oem"]; 14900f74e643SEd Tanous nlohmann::json& oemOpenbmc = oem["OpenBmc"]; 14910f74e643SEd Tanous oem["@odata.type"] = "#OemManager.Oem"; 14920f74e643SEd Tanous oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem"; 14930f74e643SEd Tanous oem["@odata.context"] = "/redfish/v1/$metadata#OemManager.Oem"; 14940f74e643SEd Tanous oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc"; 14950f74e643SEd Tanous oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc"; 14960f74e643SEd Tanous oemOpenbmc["@odata.context"] = 14970f74e643SEd Tanous "/redfish/v1/$metadata#OemManager.OpenBmc"; 14980f74e643SEd Tanous 1499ed5befbdSJennifer Lee // Update Actions object. 15000f74e643SEd Tanous nlohmann::json& manager_reset = 15010f74e643SEd Tanous res.jsonValue["Actions"]["#Manager.Reset"]; 1502ed5befbdSJennifer Lee manager_reset["target"] = 1503ed5befbdSJennifer Lee "/redfish/v1/Managers/bmc/Actions/Manager.Reset"; 1504ed5befbdSJennifer Lee manager_reset["ResetType@Redfish.AllowableValues"] = { 1505ed5befbdSJennifer Lee "GracefulRestart"}; 1506ca537928SJennifer Lee 1507cb92c03bSAndrew Geissler res.jsonValue["DateTime"] = crow::utility::dateTimeNow(); 1508474bfad5SSantosh Puranik 1509474bfad5SSantosh Puranik // Fill in GraphicalConsole and SerialConsole info 1510474bfad5SSantosh Puranik res.jsonValue["SerialConsole"]["ServiceEnabled"] = true; 1511474bfad5SSantosh Puranik res.jsonValue["SerialConsole"]["ConnectTypesSupported"] = {"IPMI", 1512474bfad5SSantosh Puranik "SSH"}; 1513474bfad5SSantosh Puranik // TODO (Santosh) : Uncomment when KVM support is in. 1514474bfad5SSantosh Puranik // res.jsonValue["GraphicalConsole"]["ServiceEnabled"] = true; 1515474bfad5SSantosh Puranik // res.jsonValue["GraphicalConsole"]["ConnectTypesSupported"] = 1516474bfad5SSantosh Puranik // {"KVMIP"}; 1517474bfad5SSantosh Puranik 1518603a6640SGunnar Mills res.jsonValue["Links"]["ManagerForServers@odata.count"] = 1; 1519603a6640SGunnar Mills res.jsonValue["Links"]["ManagerForServers"] = { 1520603a6640SGunnar Mills {{"@odata.id", "/redfish/v1/Systems/system"}}}; 152126f03899SShawn McCarney 1522ed5befbdSJennifer Lee std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res); 15235b4aa86bSJames Feist 1524ca537928SJennifer Lee crow::connections::systemBus->async_method_call( 1525ca537928SJennifer Lee [asyncResp](const boost::system::error_code ec, 15265b4aa86bSJames Feist const dbus::utility::ManagedObjectType& resp) { 1527ca537928SJennifer Lee if (ec) 1528ca537928SJennifer Lee { 1529ca537928SJennifer Lee BMCWEB_LOG_ERROR << "Error while getting Software Version"; 1530f12894f8SJason M. Bills messages::internalError(asyncResp->res); 1531ca537928SJennifer Lee return; 1532ca537928SJennifer Lee } 1533ca537928SJennifer Lee 1534ca537928SJennifer Lee for (auto& objpath : resp) 1535ca537928SJennifer Lee { 1536ca537928SJennifer Lee for (auto& interface : objpath.second) 1537ca537928SJennifer Lee { 15385f2caaefSJames Feist // If interface is 15395f2caaefSJames Feist // xyz.openbmc_project.Software.Version, this is 15405f2caaefSJames Feist // what we're looking for. 1541ca537928SJennifer Lee if (interface.first == 1542ca537928SJennifer Lee "xyz.openbmc_project.Software.Version") 1543ca537928SJennifer Lee { 1544ca537928SJennifer Lee // Cut out everyting until last "/", ... 1545ca537928SJennifer Lee for (auto& property : interface.second) 1546ca537928SJennifer Lee { 1547ca537928SJennifer Lee if (property.first == "Version") 1548ca537928SJennifer Lee { 1549ca537928SJennifer Lee const std::string* value = 1550abf2add6SEd Tanous std::get_if<std::string>( 1551abf2add6SEd Tanous &property.second); 1552ca537928SJennifer Lee if (value == nullptr) 1553ca537928SJennifer Lee { 1554ca537928SJennifer Lee continue; 1555ca537928SJennifer Lee } 1556ca537928SJennifer Lee asyncResp->res 1557ca537928SJennifer Lee .jsonValue["FirmwareVersion"] = *value; 1558ca537928SJennifer Lee } 1559ca537928SJennifer Lee } 1560ca537928SJennifer Lee } 1561ca537928SJennifer Lee } 1562ca537928SJennifer Lee } 1563ca537928SJennifer Lee }, 1564ca537928SJennifer Lee "xyz.openbmc_project.Software.BMC.Updater", 1565ca537928SJennifer Lee "/xyz/openbmc_project/software", 1566ca537928SJennifer Lee "org.freedesktop.DBus.ObjectManager", "GetManagedObjects"); 1567*73df0db0SJames Feist auto pids = std::make_shared<GetPIDValues>(asyncResp); 1568*73df0db0SJames Feist pids->run(); 156983ff9ab6SJames Feist } 15705b4aa86bSJames Feist 15715b4aa86bSJames Feist void doPatch(crow::Response& res, const crow::Request& req, 15725b4aa86bSJames Feist const std::vector<std::string>& params) override 15735b4aa86bSJames Feist { 15740627a2c7SEd Tanous std::optional<nlohmann::json> oem; 1575af5d6058SSantosh Puranik std::optional<std::string> datetime; 15760627a2c7SEd Tanous 1577af5d6058SSantosh Puranik if (!json_util::readJson(req, res, "Oem", oem, "DateTime", datetime)) 157883ff9ab6SJames Feist { 157983ff9ab6SJames Feist return; 158083ff9ab6SJames Feist } 15810627a2c7SEd Tanous 158283ff9ab6SJames Feist std::shared_ptr<AsyncResp> response = std::make_shared<AsyncResp>(res); 15830627a2c7SEd Tanous 15840627a2c7SEd Tanous if (oem) 158583ff9ab6SJames Feist { 15865f2caaefSJames Feist std::optional<nlohmann::json> openbmc; 158743b761d0SEd Tanous if (!redfish::json_util::readJson(*oem, res, "OpenBmc", openbmc)) 158883ff9ab6SJames Feist { 158943b761d0SEd Tanous BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property " 159043b761d0SEd Tanous << oem->dump(); 159183ff9ab6SJames Feist return; 159283ff9ab6SJames Feist } 15935f2caaefSJames Feist if (openbmc) 159483ff9ab6SJames Feist { 15955f2caaefSJames Feist std::optional<nlohmann::json> fan; 159643b761d0SEd Tanous if (!redfish::json_util::readJson(*openbmc, res, "Fan", fan)) 159783ff9ab6SJames Feist { 15985f2caaefSJames Feist BMCWEB_LOG_ERROR << "Line:" << __LINE__ 15995f2caaefSJames Feist << ", Illegal Property " 16005f2caaefSJames Feist << openbmc->dump(); 160183ff9ab6SJames Feist return; 160283ff9ab6SJames Feist } 16035f2caaefSJames Feist if (fan) 160483ff9ab6SJames Feist { 1605*73df0db0SJames Feist auto pid = std::make_shared<SetPIDValues>(response, *fan); 1606*73df0db0SJames Feist pid->run(); 160783ff9ab6SJames Feist } 160883ff9ab6SJames Feist } 160983ff9ab6SJames Feist } 1610af5d6058SSantosh Puranik if (datetime) 1611af5d6058SSantosh Puranik { 1612af5d6058SSantosh Puranik setDateTime(response, std::move(*datetime)); 1613af5d6058SSantosh Puranik } 1614af5d6058SSantosh Puranik } 1615af5d6058SSantosh Puranik 1616af5d6058SSantosh Puranik void setDateTime(std::shared_ptr<AsyncResp> aResp, 1617af5d6058SSantosh Puranik std::string datetime) const 1618af5d6058SSantosh Puranik { 1619af5d6058SSantosh Puranik BMCWEB_LOG_DEBUG << "Set date time: " << datetime; 1620af5d6058SSantosh Puranik 1621af5d6058SSantosh Puranik std::stringstream stream(datetime); 1622af5d6058SSantosh Puranik // Convert from ISO 8601 to boost local_time 1623af5d6058SSantosh Puranik // (BMC only has time in UTC) 1624af5d6058SSantosh Puranik boost::posix_time::ptime posixTime; 1625af5d6058SSantosh Puranik boost::posix_time::ptime epoch(boost::gregorian::date(1970, 1, 1)); 1626af5d6058SSantosh Puranik // Facet gets deleted with the stringsteam 1627af5d6058SSantosh Puranik auto ifc = std::make_unique<boost::local_time::local_time_input_facet>( 1628af5d6058SSantosh Puranik "%Y-%m-%d %H:%M:%S%F %ZP"); 1629af5d6058SSantosh Puranik stream.imbue(std::locale(stream.getloc(), ifc.release())); 1630af5d6058SSantosh Puranik 1631af5d6058SSantosh Puranik boost::local_time::local_date_time ldt( 1632af5d6058SSantosh Puranik boost::local_time::not_a_date_time); 1633af5d6058SSantosh Puranik 1634af5d6058SSantosh Puranik if (stream >> ldt) 1635af5d6058SSantosh Puranik { 1636af5d6058SSantosh Puranik posixTime = ldt.utc_time(); 1637af5d6058SSantosh Puranik boost::posix_time::time_duration dur = posixTime - epoch; 1638af5d6058SSantosh Puranik uint64_t durMicroSecs = 1639af5d6058SSantosh Puranik static_cast<uint64_t>(dur.total_microseconds()); 1640af5d6058SSantosh Puranik crow::connections::systemBus->async_method_call( 1641af5d6058SSantosh Puranik [aResp{std::move(aResp)}, datetime{std::move(datetime)}]( 1642af5d6058SSantosh Puranik const boost::system::error_code ec) { 1643af5d6058SSantosh Puranik if (ec) 1644af5d6058SSantosh Puranik { 1645af5d6058SSantosh Puranik BMCWEB_LOG_DEBUG << "Failed to set elapsed time. " 1646af5d6058SSantosh Puranik "DBUS response error " 1647af5d6058SSantosh Puranik << ec; 1648af5d6058SSantosh Puranik messages::internalError(aResp->res); 1649af5d6058SSantosh Puranik return; 1650af5d6058SSantosh Puranik } 1651af5d6058SSantosh Puranik aResp->res.jsonValue["DateTime"] = datetime; 1652af5d6058SSantosh Puranik }, 1653af5d6058SSantosh Puranik "xyz.openbmc_project.Time.Manager", 1654af5d6058SSantosh Puranik "/xyz/openbmc_project/time/bmc", 1655af5d6058SSantosh Puranik "org.freedesktop.DBus.Properties", "Set", 1656af5d6058SSantosh Puranik "xyz.openbmc_project.Time.EpochTime", "Elapsed", 1657af5d6058SSantosh Puranik std::variant<uint64_t>(durMicroSecs)); 1658af5d6058SSantosh Puranik } 1659af5d6058SSantosh Puranik else 1660af5d6058SSantosh Puranik { 1661af5d6058SSantosh Puranik messages::propertyValueFormatError(aResp->res, datetime, 1662af5d6058SSantosh Puranik "DateTime"); 1663af5d6058SSantosh Puranik return; 1664af5d6058SSantosh Puranik } 166583ff9ab6SJames Feist } 16669c310685SBorawski.Lukasz 16670f74e643SEd Tanous std::string uuid; 16689c310685SBorawski.Lukasz }; 16699c310685SBorawski.Lukasz 16701abe55efSEd Tanous class ManagerCollection : public Node 16711abe55efSEd Tanous { 16729c310685SBorawski.Lukasz public: 16731abe55efSEd Tanous ManagerCollection(CrowApp& app) : Node(app, "/redfish/v1/Managers/") 16741abe55efSEd Tanous { 1675a434f2bdSEd Tanous entityPrivileges = { 1676a434f2bdSEd Tanous {boost::beast::http::verb::get, {{"Login"}}}, 1677e0d918bcSEd Tanous {boost::beast::http::verb::head, {{"Login"}}}, 1678e0d918bcSEd Tanous {boost::beast::http::verb::patch, {{"ConfigureManager"}}}, 1679e0d918bcSEd Tanous {boost::beast::http::verb::put, {{"ConfigureManager"}}}, 1680e0d918bcSEd Tanous {boost::beast::http::verb::delete_, {{"ConfigureManager"}}}, 1681e0d918bcSEd Tanous {boost::beast::http::verb::post, {{"ConfigureManager"}}}}; 16829c310685SBorawski.Lukasz } 16839c310685SBorawski.Lukasz 16849c310685SBorawski.Lukasz private: 168555c7b7a2SEd Tanous void doGet(crow::Response& res, const crow::Request& req, 16861abe55efSEd Tanous const std::vector<std::string>& params) override 16871abe55efSEd Tanous { 168883ff9ab6SJames Feist // Collections don't include the static data added by SubRoute 168983ff9ab6SJames Feist // because it has a duplicate entry for members 169055c7b7a2SEd Tanous res.jsonValue["@odata.id"] = "/redfish/v1/Managers"; 169155c7b7a2SEd Tanous res.jsonValue["@odata.type"] = "#ManagerCollection.ManagerCollection"; 169255c7b7a2SEd Tanous res.jsonValue["@odata.context"] = 169355c7b7a2SEd Tanous "/redfish/v1/$metadata#ManagerCollection.ManagerCollection"; 169455c7b7a2SEd Tanous res.jsonValue["Name"] = "Manager Collection"; 169555c7b7a2SEd Tanous res.jsonValue["Members@odata.count"] = 1; 169655c7b7a2SEd Tanous res.jsonValue["Members"] = { 16975b4aa86bSJames Feist {{"@odata.id", "/redfish/v1/Managers/bmc"}}}; 16989c310685SBorawski.Lukasz res.end(); 16999c310685SBorawski.Lukasz } 17009c310685SBorawski.Lukasz }; 17019c310685SBorawski.Lukasz } // namespace redfish 1702