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> 215b4aa86bSJames Feist #include <dbus_utility.hpp> 225b4aa86bSJames Feist 231abe55efSEd Tanous namespace redfish 241abe55efSEd Tanous { 25ed5befbdSJennifer Lee 26ed5befbdSJennifer Lee /** 27ed5befbdSJennifer Lee * ManagerActionsReset class supports handle POST method for Reset action. 28ed5befbdSJennifer Lee * The class retrieves and sends data directly to dbus. 29ed5befbdSJennifer Lee */ 30ed5befbdSJennifer Lee class ManagerActionsReset : public Node 31ed5befbdSJennifer Lee { 32ed5befbdSJennifer Lee public: 33ed5befbdSJennifer Lee ManagerActionsReset(CrowApp& app) : 34ed5befbdSJennifer Lee Node(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/") 35ed5befbdSJennifer Lee { 36ed5befbdSJennifer Lee entityPrivileges = { 37ed5befbdSJennifer Lee {boost::beast::http::verb::get, {{"Login"}}}, 38ed5befbdSJennifer Lee {boost::beast::http::verb::head, {{"Login"}}}, 39ed5befbdSJennifer Lee {boost::beast::http::verb::patch, {{"ConfigureManager"}}}, 40ed5befbdSJennifer Lee {boost::beast::http::verb::put, {{"ConfigureManager"}}}, 41ed5befbdSJennifer Lee {boost::beast::http::verb::delete_, {{"ConfigureManager"}}}, 42ed5befbdSJennifer Lee {boost::beast::http::verb::post, {{"ConfigureManager"}}}}; 43ed5befbdSJennifer Lee } 44ed5befbdSJennifer Lee 45ed5befbdSJennifer Lee private: 46ed5befbdSJennifer Lee /** 47ed5befbdSJennifer Lee * Function handles POST method request. 48ed5befbdSJennifer Lee * Analyzes POST body message before sends Reset request data to dbus. 49ed5befbdSJennifer Lee * OpenBMC allows for ResetType is GracefulRestart only. 50ed5befbdSJennifer Lee */ 51ed5befbdSJennifer Lee void doPost(crow::Response& res, const crow::Request& req, 52ed5befbdSJennifer Lee const std::vector<std::string>& params) override 53ed5befbdSJennifer Lee { 54ed5befbdSJennifer Lee std::string resetType; 55ed5befbdSJennifer Lee 56ed5befbdSJennifer Lee if (!json_util::readJson(req, res, "ResetType", resetType)) 57ed5befbdSJennifer Lee { 58ed5befbdSJennifer Lee return; 59ed5befbdSJennifer Lee } 60ed5befbdSJennifer Lee 61ed5befbdSJennifer Lee if (resetType != "GracefulRestart") 62ed5befbdSJennifer Lee { 63ed5befbdSJennifer Lee res.result(boost::beast::http::status::bad_request); 64ed5befbdSJennifer Lee messages::actionParameterNotSupported(res, resetType, "ResetType"); 65ed5befbdSJennifer Lee BMCWEB_LOG_ERROR << "Request incorrect action parameter: " 66ed5befbdSJennifer Lee << resetType; 67ed5befbdSJennifer Lee res.end(); 68ed5befbdSJennifer Lee return; 69ed5befbdSJennifer Lee } 70ed5befbdSJennifer Lee doBMCGracefulRestart(res, req, params); 71ed5befbdSJennifer Lee } 72ed5befbdSJennifer Lee 73ed5befbdSJennifer Lee /** 74ed5befbdSJennifer Lee * Function transceives data with dbus directly. 75ed5befbdSJennifer Lee * All BMC state properties will be retrieved before sending reset request. 76ed5befbdSJennifer Lee */ 77ed5befbdSJennifer Lee void doBMCGracefulRestart(crow::Response& res, const crow::Request& req, 78ed5befbdSJennifer Lee const std::vector<std::string>& params) 79ed5befbdSJennifer Lee { 80ed5befbdSJennifer Lee const char* processName = "xyz.openbmc_project.State.BMC"; 81ed5befbdSJennifer Lee const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 82ed5befbdSJennifer Lee const char* interfaceName = "xyz.openbmc_project.State.BMC"; 83ed5befbdSJennifer Lee const std::string& propertyValue = 84ed5befbdSJennifer Lee "xyz.openbmc_project.State.BMC.Transition.Reboot"; 85ed5befbdSJennifer Lee const char* destProperty = "RequestedBMCTransition"; 86ed5befbdSJennifer Lee 87ed5befbdSJennifer Lee // Create the D-Bus variant for D-Bus call. 88ed5befbdSJennifer Lee VariantType dbusPropertyValue(propertyValue); 89ed5befbdSJennifer Lee 90ed5befbdSJennifer Lee std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res); 91ed5befbdSJennifer Lee 92ed5befbdSJennifer Lee crow::connections::systemBus->async_method_call( 93ed5befbdSJennifer Lee [asyncResp](const boost::system::error_code ec) { 94ed5befbdSJennifer Lee // Use "Set" method to set the property value. 95ed5befbdSJennifer Lee if (ec) 96ed5befbdSJennifer Lee { 97ed5befbdSJennifer Lee BMCWEB_LOG_ERROR << "[Set] Bad D-Bus request error: " << ec; 98ed5befbdSJennifer Lee messages::internalError(asyncResp->res); 99ed5befbdSJennifer Lee return; 100ed5befbdSJennifer Lee } 101ed5befbdSJennifer Lee 102ed5befbdSJennifer Lee messages::success(asyncResp->res); 103ed5befbdSJennifer Lee }, 104ed5befbdSJennifer Lee processName, objectPath, "org.freedesktop.DBus.Properties", "Set", 105ed5befbdSJennifer Lee interfaceName, destProperty, dbusPropertyValue); 106ed5befbdSJennifer Lee } 107ed5befbdSJennifer Lee }; 108ed5befbdSJennifer Lee 1095b4aa86bSJames Feist static constexpr const char* objectManagerIface = 1105b4aa86bSJames Feist "org.freedesktop.DBus.ObjectManager"; 1115b4aa86bSJames Feist static constexpr const char* pidConfigurationIface = 1125b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid"; 1135b4aa86bSJames Feist static constexpr const char* pidZoneConfigurationIface = 1145b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid.Zone"; 1159c310685SBorawski.Lukasz 1165b4aa86bSJames Feist static void asyncPopulatePid(const std::string& connection, 1175b4aa86bSJames Feist const std::string& path, 1185b4aa86bSJames Feist std::shared_ptr<AsyncResp> asyncResp) 1195b4aa86bSJames Feist { 1205b4aa86bSJames Feist 1215b4aa86bSJames Feist crow::connections::systemBus->async_method_call( 1225b4aa86bSJames Feist [asyncResp](const boost::system::error_code ec, 1235b4aa86bSJames Feist const dbus::utility::ManagedObjectType& managedObj) { 1245b4aa86bSJames Feist if (ec) 1255b4aa86bSJames Feist { 1265b4aa86bSJames Feist BMCWEB_LOG_ERROR << ec; 1275b4aa86bSJames Feist asyncResp->res.jsonValue.clear(); 128f12894f8SJason M. Bills messages::internalError(asyncResp->res); 1295b4aa86bSJames Feist return; 1305b4aa86bSJames Feist } 1315b4aa86bSJames Feist nlohmann::json& configRoot = 1325b4aa86bSJames Feist asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"]; 1335b4aa86bSJames Feist nlohmann::json& fans = configRoot["FanControllers"]; 1345b4aa86bSJames Feist fans["@odata.type"] = "#OemManager.FanControllers"; 1355b4aa86bSJames Feist fans["@odata.context"] = 1365b4aa86bSJames Feist "/redfish/v1/$metadata#OemManager.FanControllers"; 1375b4aa86bSJames Feist fans["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/" 1385b4aa86bSJames Feist "Fan/FanControllers"; 1395b4aa86bSJames Feist 1405b4aa86bSJames Feist nlohmann::json& pids = configRoot["PidControllers"]; 1415b4aa86bSJames Feist pids["@odata.type"] = "#OemManager.PidControllers"; 1425b4aa86bSJames Feist pids["@odata.context"] = 1435b4aa86bSJames Feist "/redfish/v1/$metadata#OemManager.PidControllers"; 1445b4aa86bSJames Feist pids["@odata.id"] = 1455b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers"; 1465b4aa86bSJames Feist 1475b4aa86bSJames Feist nlohmann::json& zones = configRoot["FanZones"]; 1485b4aa86bSJames Feist zones["@odata.id"] = 1495b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones"; 1505b4aa86bSJames Feist zones["@odata.type"] = "#OemManager.FanZones"; 1515b4aa86bSJames Feist zones["@odata.context"] = 1525b4aa86bSJames Feist "/redfish/v1/$metadata#OemManager.FanZones"; 1535b4aa86bSJames Feist configRoot["@odata.id"] = 1545b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan"; 1555b4aa86bSJames Feist configRoot["@odata.type"] = "#OemManager.Fan"; 1565b4aa86bSJames Feist configRoot["@odata.context"] = 1575b4aa86bSJames Feist "/redfish/v1/$metadata#OemManager.Fan"; 1585b4aa86bSJames Feist 1595b4aa86bSJames Feist bool propertyError = false; 1605b4aa86bSJames Feist for (const auto& pathPair : managedObj) 1615b4aa86bSJames Feist { 1625b4aa86bSJames Feist for (const auto& intfPair : pathPair.second) 1635b4aa86bSJames Feist { 1645b4aa86bSJames Feist if (intfPair.first != pidConfigurationIface && 1655b4aa86bSJames Feist intfPair.first != pidZoneConfigurationIface) 1665b4aa86bSJames Feist { 1675b4aa86bSJames Feist continue; 1685b4aa86bSJames Feist } 1695b4aa86bSJames Feist auto findName = intfPair.second.find("Name"); 1705b4aa86bSJames Feist if (findName == intfPair.second.end()) 1715b4aa86bSJames Feist { 1725b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Field missing Name"; 173a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 1745b4aa86bSJames Feist return; 1755b4aa86bSJames Feist } 1765b4aa86bSJames Feist const std::string* namePtr = 1775b4aa86bSJames Feist mapbox::getPtr<const std::string>(findName->second); 1785b4aa86bSJames Feist if (namePtr == nullptr) 1795b4aa86bSJames Feist { 1805b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Name Field illegal"; 1815b4aa86bSJames Feist return; 1825b4aa86bSJames Feist } 1835b4aa86bSJames Feist 1845b4aa86bSJames Feist std::string name = *namePtr; 1855b4aa86bSJames Feist dbus::utility::escapePathForDbus(name); 1865b4aa86bSJames Feist if (intfPair.first == pidZoneConfigurationIface) 1875b4aa86bSJames Feist { 1885b4aa86bSJames Feist std::string chassis; 1895b4aa86bSJames Feist if (!dbus::utility::getNthStringFromPath( 1905b4aa86bSJames Feist pathPair.first.str, 5, chassis)) 1915b4aa86bSJames Feist { 1925b4aa86bSJames Feist chassis = "#IllegalValue"; 1935b4aa86bSJames Feist } 1945b4aa86bSJames Feist nlohmann::json& zone = zones[name]; 1955b4aa86bSJames Feist zone["Chassis"] = { 1965b4aa86bSJames Feist {"@odata.id", "/redfish/v1/Chassis/" + chassis}}; 1975b4aa86bSJames Feist zone["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/" 1985b4aa86bSJames Feist "OpenBmc/Fan/FanZones/" + 1995b4aa86bSJames Feist name; 2005b4aa86bSJames Feist zone["@odata.type"] = "#OemManager.FanZone"; 2015b4aa86bSJames Feist zone["@odata.context"] = 2025b4aa86bSJames Feist "/redfish/v1/$metadata#OemManager.FanZone"; 2035b4aa86bSJames Feist } 2045b4aa86bSJames Feist 2055b4aa86bSJames Feist for (const auto& propertyPair : intfPair.second) 2065b4aa86bSJames Feist { 2075b4aa86bSJames Feist if (propertyPair.first == "Type" || 2085b4aa86bSJames Feist propertyPair.first == "Class" || 2095b4aa86bSJames Feist propertyPair.first == "Name") 2105b4aa86bSJames Feist { 2115b4aa86bSJames Feist continue; 2125b4aa86bSJames Feist } 2135b4aa86bSJames Feist 2145b4aa86bSJames Feist // zones 2155b4aa86bSJames Feist if (intfPair.first == pidZoneConfigurationIface) 2165b4aa86bSJames Feist { 2175b4aa86bSJames Feist const double* ptr = mapbox::getPtr<const double>( 2185b4aa86bSJames Feist propertyPair.second); 2195b4aa86bSJames Feist if (ptr == nullptr) 2205b4aa86bSJames Feist { 2215b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 2225b4aa86bSJames Feist << propertyPair.first; 223f12894f8SJason M. Bills messages::internalError(asyncResp->res); 2245b4aa86bSJames Feist return; 2255b4aa86bSJames Feist } 2265b4aa86bSJames Feist zones[name][propertyPair.first] = *ptr; 2275b4aa86bSJames Feist } 2285b4aa86bSJames Feist 2295b4aa86bSJames Feist // pid and fans are off the same configuration 2305b4aa86bSJames Feist if (intfPair.first == pidConfigurationIface) 2315b4aa86bSJames Feist { 2325b4aa86bSJames Feist const std::string* classPtr = nullptr; 2335b4aa86bSJames Feist auto findClass = intfPair.second.find("Class"); 2345b4aa86bSJames Feist if (findClass != intfPair.second.end()) 2355b4aa86bSJames Feist { 2365b4aa86bSJames Feist classPtr = mapbox::getPtr<const std::string>( 2375b4aa86bSJames Feist findClass->second); 2385b4aa86bSJames Feist } 2395b4aa86bSJames Feist if (classPtr == nullptr) 2405b4aa86bSJames Feist { 2415b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Pid Class Field illegal"; 242a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 2435b4aa86bSJames Feist return; 2445b4aa86bSJames Feist } 2455b4aa86bSJames Feist bool isFan = *classPtr == "fan"; 2465b4aa86bSJames Feist nlohmann::json& element = 2475b4aa86bSJames Feist isFan ? fans[name] : pids[name]; 2485b4aa86bSJames Feist if (isFan) 2495b4aa86bSJames Feist { 2505b4aa86bSJames Feist element["@odata.id"] = 2515b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/" 2525b4aa86bSJames Feist "OpenBmc/Fan/FanControllers/" + 2535b4aa86bSJames Feist std::string(name); 2545b4aa86bSJames Feist element["@odata.type"] = 2555b4aa86bSJames Feist "#OemManager.FanController"; 2565b4aa86bSJames Feist 2575b4aa86bSJames Feist element["@odata.context"] = 2585b4aa86bSJames Feist "/redfish/v1/" 2595b4aa86bSJames Feist "$metadata#OemManager.FanController"; 2605b4aa86bSJames Feist } 2615b4aa86bSJames Feist else 2625b4aa86bSJames Feist { 2635b4aa86bSJames Feist element["@odata.id"] = 2645b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/" 2655b4aa86bSJames Feist "OpenBmc/Fan/PidControllers/" + 2665b4aa86bSJames Feist std::string(name); 2675b4aa86bSJames Feist element["@odata.type"] = 2685b4aa86bSJames Feist "#OemManager.PidController"; 2695b4aa86bSJames Feist element["@odata.context"] = 2705b4aa86bSJames Feist "/redfish/v1/$metadata" 2715b4aa86bSJames Feist "#OemManager.PidController"; 2725b4aa86bSJames Feist } 2735b4aa86bSJames Feist 2745b4aa86bSJames Feist if (propertyPair.first == "Zones") 2755b4aa86bSJames Feist { 2765b4aa86bSJames Feist const std::vector<std::string>* inputs = 2775b4aa86bSJames Feist mapbox::getPtr< 2785b4aa86bSJames Feist const std::vector<std::string>>( 2795b4aa86bSJames Feist propertyPair.second); 2805b4aa86bSJames Feist 2815b4aa86bSJames Feist if (inputs == nullptr) 2825b4aa86bSJames Feist { 2835b4aa86bSJames Feist BMCWEB_LOG_ERROR 2845b4aa86bSJames Feist << "Zones Pid Field Illegal"; 285a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 2865b4aa86bSJames Feist return; 2875b4aa86bSJames Feist } 2885b4aa86bSJames Feist auto& data = element[propertyPair.first]; 2895b4aa86bSJames Feist data = nlohmann::json::array(); 2905b4aa86bSJames Feist for (std::string itemCopy : *inputs) 2915b4aa86bSJames Feist { 2925b4aa86bSJames Feist dbus::utility::escapePathForDbus(itemCopy); 2935b4aa86bSJames Feist data.push_back( 2945b4aa86bSJames Feist {{"@odata.id", 2955b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/" 2965b4aa86bSJames Feist "OpenBmc/Fan/FanZones/" + 2975b4aa86bSJames Feist itemCopy}}); 2985b4aa86bSJames Feist } 2995b4aa86bSJames Feist } 3005b4aa86bSJames Feist // todo(james): may never happen, but this 3015b4aa86bSJames Feist // assumes configuration data referenced in the 3025b4aa86bSJames Feist // PID config is provided by the same daemon, we 3035b4aa86bSJames Feist // could add another loop to cover all cases, 3045b4aa86bSJames Feist // but I'm okay kicking this can down the road a 3055b4aa86bSJames Feist // bit 3065b4aa86bSJames Feist 3075b4aa86bSJames Feist else if (propertyPair.first == "Inputs" || 3085b4aa86bSJames Feist propertyPair.first == "Outputs") 3095b4aa86bSJames Feist { 3105b4aa86bSJames Feist auto& data = element[propertyPair.first]; 3115b4aa86bSJames Feist const std::vector<std::string>* inputs = 3125b4aa86bSJames Feist mapbox::getPtr< 3135b4aa86bSJames Feist const std::vector<std::string>>( 3145b4aa86bSJames Feist propertyPair.second); 3155b4aa86bSJames Feist 3165b4aa86bSJames Feist if (inputs == nullptr) 3175b4aa86bSJames Feist { 3185b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 3195b4aa86bSJames Feist << propertyPair.first; 320f12894f8SJason M. Bills messages::internalError(asyncResp->res); 3215b4aa86bSJames Feist return; 3225b4aa86bSJames Feist } 3235b4aa86bSJames Feist data = *inputs; 3245b4aa86bSJames Feist } // doubles 3255b4aa86bSJames Feist else if (propertyPair.first == 3265b4aa86bSJames Feist "FFGainCoefficient" || 3275b4aa86bSJames Feist propertyPair.first == "FFOffCoefficient" || 3285b4aa86bSJames Feist propertyPair.first == "ICoefficient" || 3295b4aa86bSJames Feist propertyPair.first == "ILimitMax" || 3305b4aa86bSJames Feist propertyPair.first == "ILimitMin" || 3315b4aa86bSJames Feist propertyPair.first == "OutLimitMax" || 3325b4aa86bSJames Feist propertyPair.first == "OutLimitMin" || 3335b4aa86bSJames Feist propertyPair.first == "PCoefficient" || 3345b4aa86bSJames Feist propertyPair.first == "SlewNeg" || 3355b4aa86bSJames Feist propertyPair.first == "SlewPos") 3365b4aa86bSJames Feist { 3375b4aa86bSJames Feist const double* ptr = 3385b4aa86bSJames Feist mapbox::getPtr<const double>( 3395b4aa86bSJames Feist propertyPair.second); 3405b4aa86bSJames Feist if (ptr == nullptr) 3415b4aa86bSJames Feist { 3425b4aa86bSJames Feist BMCWEB_LOG_ERROR << "Field Illegal " 3435b4aa86bSJames Feist << propertyPair.first; 344f12894f8SJason M. Bills messages::internalError(asyncResp->res); 3455b4aa86bSJames Feist return; 3465b4aa86bSJames Feist } 3475b4aa86bSJames Feist element[propertyPair.first] = *ptr; 3485b4aa86bSJames Feist } 3495b4aa86bSJames Feist } 3505b4aa86bSJames Feist } 3515b4aa86bSJames Feist } 3525b4aa86bSJames Feist } 3535b4aa86bSJames Feist }, 3545b4aa86bSJames Feist connection, path, objectManagerIface, "GetManagedObjects"); 3555b4aa86bSJames Feist } 356ca537928SJennifer Lee 35783ff9ab6SJames Feist enum class CreatePIDRet 35883ff9ab6SJames Feist { 35983ff9ab6SJames Feist fail, 36083ff9ab6SJames Feist del, 36183ff9ab6SJames Feist patch 36283ff9ab6SJames Feist }; 36383ff9ab6SJames Feist 36483ff9ab6SJames Feist static CreatePIDRet createPidInterface( 36583ff9ab6SJames Feist const std::shared_ptr<AsyncResp>& response, const std::string& type, 36683ff9ab6SJames Feist const nlohmann::json& record, const std::string& path, 36783ff9ab6SJames Feist const dbus::utility::ManagedObjectType& managedObj, bool createNewObject, 36883ff9ab6SJames Feist boost::container::flat_map<std::string, dbus::utility::DbusVariantType>& 36983ff9ab6SJames Feist output, 37083ff9ab6SJames Feist std::string& chassis) 37183ff9ab6SJames Feist { 37283ff9ab6SJames Feist 37383ff9ab6SJames Feist if (type == "PidControllers" || type == "FanControllers") 37483ff9ab6SJames Feist { 37583ff9ab6SJames Feist if (createNewObject) 37683ff9ab6SJames Feist { 37783ff9ab6SJames Feist output["Class"] = type == "PidControllers" ? std::string("temp") 37883ff9ab6SJames Feist : std::string("fan"); 37983ff9ab6SJames Feist output["Type"] = std::string("Pid"); 38083ff9ab6SJames Feist } 38183ff9ab6SJames Feist else if (record == nullptr) 38283ff9ab6SJames Feist { 38383ff9ab6SJames Feist // delete interface 38483ff9ab6SJames Feist crow::connections::systemBus->async_method_call( 38583ff9ab6SJames Feist [response, 38683ff9ab6SJames Feist path{std::string(path)}](const boost::system::error_code ec) { 38783ff9ab6SJames Feist if (ec) 38883ff9ab6SJames Feist { 38983ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Error patching " << path << ": " 39083ff9ab6SJames Feist << ec; 39135a62c7cSJason M. Bills messages::internalError(response->res); 39283ff9ab6SJames Feist } 39383ff9ab6SJames Feist }, 39483ff9ab6SJames Feist "xyz.openbmc_project.EntityManager", path, 39583ff9ab6SJames Feist pidConfigurationIface, "Delete"); 39683ff9ab6SJames Feist return CreatePIDRet::del; 39783ff9ab6SJames Feist } 39883ff9ab6SJames Feist 39983ff9ab6SJames Feist for (auto& field : record.items()) 40083ff9ab6SJames Feist { 40183ff9ab6SJames Feist if (field.key() == "Zones") 40283ff9ab6SJames Feist { 40383ff9ab6SJames Feist if (!field.value().is_array()) 40483ff9ab6SJames Feist { 40583ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Illegal Type " << field.key(); 40635a62c7cSJason M. Bills messages::propertyValueFormatError( 40735a62c7cSJason M. Bills response->res, field.value(), field.key()); 40883ff9ab6SJames Feist return CreatePIDRet::fail; 40983ff9ab6SJames Feist } 41083ff9ab6SJames Feist std::vector<std::string> inputs; 41183ff9ab6SJames Feist for (const auto& odata : field.value().items()) 41283ff9ab6SJames Feist { 41383ff9ab6SJames Feist for (const auto& value : odata.value().items()) 41483ff9ab6SJames Feist { 41583ff9ab6SJames Feist const std::string* path = 41683ff9ab6SJames Feist value.value().get_ptr<const std::string*>(); 41783ff9ab6SJames Feist if (path == nullptr) 41883ff9ab6SJames Feist { 41983ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Illegal Type " << field.key(); 42083ff9ab6SJames Feist messages::propertyValueFormatError( 42135a62c7cSJason M. Bills response->res, field.value().dump(), 42235a62c7cSJason M. Bills field.key()); 42383ff9ab6SJames Feist return CreatePIDRet::fail; 42483ff9ab6SJames Feist } 42583ff9ab6SJames Feist std::string input; 42683ff9ab6SJames Feist if (!dbus::utility::getNthStringFromPath(*path, 4, 42783ff9ab6SJames Feist input)) 42883ff9ab6SJames Feist { 42983ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Got invalid path " << *path; 43083ff9ab6SJames Feist messages::propertyValueFormatError( 43135a62c7cSJason M. Bills response->res, field.value().dump(), 43235a62c7cSJason M. Bills field.key()); 43383ff9ab6SJames Feist return CreatePIDRet::fail; 43483ff9ab6SJames Feist } 43583ff9ab6SJames Feist boost::replace_all(input, "_", " "); 43683ff9ab6SJames Feist inputs.emplace_back(std::move(input)); 43783ff9ab6SJames Feist } 43883ff9ab6SJames Feist } 43983ff9ab6SJames Feist output["Zones"] = std::move(inputs); 44083ff9ab6SJames Feist } 44183ff9ab6SJames Feist else if (field.key() == "Inputs" || field.key() == "Outputs") 44283ff9ab6SJames Feist { 44383ff9ab6SJames Feist if (!field.value().is_array()) 44483ff9ab6SJames Feist { 44583ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Illegal Type " << field.key(); 44635a62c7cSJason M. Bills messages::propertyValueFormatError( 44735a62c7cSJason M. Bills response->res, field.value().dump(), field.key()); 44883ff9ab6SJames Feist return CreatePIDRet::fail; 44983ff9ab6SJames Feist } 45083ff9ab6SJames Feist std::vector<std::string> inputs; 45183ff9ab6SJames Feist for (const auto& value : field.value().items()) 45283ff9ab6SJames Feist { 45383ff9ab6SJames Feist const std::string* sensor = 45483ff9ab6SJames Feist value.value().get_ptr<const std::string*>(); 45583ff9ab6SJames Feist 45683ff9ab6SJames Feist if (sensor == nullptr) 45783ff9ab6SJames Feist { 45883ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Illegal Type " 45983ff9ab6SJames Feist << field.value().dump(); 46083ff9ab6SJames Feist messages::propertyValueFormatError( 46135a62c7cSJason M. Bills response->res, field.value().dump(), field.key()); 46283ff9ab6SJames Feist return CreatePIDRet::fail; 46383ff9ab6SJames Feist } 46483ff9ab6SJames Feist 46583ff9ab6SJames Feist std::string input = 46683ff9ab6SJames Feist boost::replace_all_copy(*sensor, "_", " "); 46783ff9ab6SJames Feist inputs.push_back(std::move(input)); 46883ff9ab6SJames Feist // try to find the sensor in the 46983ff9ab6SJames Feist // configuration 47083ff9ab6SJames Feist if (chassis.empty()) 47183ff9ab6SJames Feist { 47283ff9ab6SJames Feist std::find_if( 47383ff9ab6SJames Feist managedObj.begin(), managedObj.end(), 47483ff9ab6SJames Feist [&chassis, sensor](const auto& obj) { 47583ff9ab6SJames Feist if (boost::algorithm::ends_with(obj.first.str, 47683ff9ab6SJames Feist *sensor)) 47783ff9ab6SJames Feist { 47883ff9ab6SJames Feist return dbus::utility::getNthStringFromPath( 47983ff9ab6SJames Feist obj.first.str, 5, chassis); 48083ff9ab6SJames Feist } 48183ff9ab6SJames Feist return false; 48283ff9ab6SJames Feist }); 48383ff9ab6SJames Feist } 48483ff9ab6SJames Feist } 48583ff9ab6SJames Feist output[field.key()] = inputs; 48683ff9ab6SJames Feist } 48783ff9ab6SJames Feist 48883ff9ab6SJames Feist // doubles 48983ff9ab6SJames Feist else if (field.key() == "FFGainCoefficient" || 49083ff9ab6SJames Feist field.key() == "FFOffCoefficient" || 49183ff9ab6SJames Feist field.key() == "ICoefficient" || 49283ff9ab6SJames Feist field.key() == "ILimitMax" || field.key() == "ILimitMin" || 49383ff9ab6SJames Feist field.key() == "OutLimitMax" || 49483ff9ab6SJames Feist field.key() == "OutLimitMin" || 49583ff9ab6SJames Feist field.key() == "PCoefficient" || 49683ff9ab6SJames Feist field.key() == "SetPoint" || field.key() == "SlewNeg" || 49783ff9ab6SJames Feist field.key() == "SlewPos") 49883ff9ab6SJames Feist { 49983ff9ab6SJames Feist const double* ptr = field.value().get_ptr<const double*>(); 50083ff9ab6SJames Feist if (ptr == nullptr) 50183ff9ab6SJames Feist { 50283ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Illegal Type " << field.key(); 50335a62c7cSJason M. Bills messages::propertyValueFormatError( 50435a62c7cSJason M. Bills response->res, field.value().dump(), field.key()); 50583ff9ab6SJames Feist return CreatePIDRet::fail; 50683ff9ab6SJames Feist } 50783ff9ab6SJames Feist output[field.key()] = *ptr; 50883ff9ab6SJames Feist } 50983ff9ab6SJames Feist 51083ff9ab6SJames Feist else 51183ff9ab6SJames Feist { 51283ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Illegal Type " << field.key(); 51335a62c7cSJason M. Bills messages::propertyUnknown(response->res, field.key()); 51483ff9ab6SJames Feist return CreatePIDRet::fail; 51583ff9ab6SJames Feist } 51683ff9ab6SJames Feist } 51783ff9ab6SJames Feist } 51883ff9ab6SJames Feist else if (type == "FanZones") 51983ff9ab6SJames Feist { 52083ff9ab6SJames Feist if (!createNewObject && record == nullptr) 52183ff9ab6SJames Feist { 52283ff9ab6SJames Feist // delete interface 52383ff9ab6SJames Feist crow::connections::systemBus->async_method_call( 52483ff9ab6SJames Feist [response, 52583ff9ab6SJames Feist path{std::string(path)}](const boost::system::error_code ec) { 52683ff9ab6SJames Feist if (ec) 52783ff9ab6SJames Feist { 52883ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Error patching " << path << ": " 52983ff9ab6SJames Feist << ec; 53035a62c7cSJason M. Bills messages::internalError(response->res); 53183ff9ab6SJames Feist } 53283ff9ab6SJames Feist }, 53383ff9ab6SJames Feist "xyz.openbmc_project.EntityManager", path, 53483ff9ab6SJames Feist pidZoneConfigurationIface, "Delete"); 53583ff9ab6SJames Feist return CreatePIDRet::del; 53683ff9ab6SJames Feist } 53783ff9ab6SJames Feist output["Type"] = std::string("Pid.Zone"); 53883ff9ab6SJames Feist 53983ff9ab6SJames Feist for (auto& field : record.items()) 54083ff9ab6SJames Feist { 54183ff9ab6SJames Feist if (field.key() == "Chassis") 54283ff9ab6SJames Feist { 54383ff9ab6SJames Feist const std::string* chassisId = nullptr; 54483ff9ab6SJames Feist for (const auto& id : field.value().items()) 54583ff9ab6SJames Feist { 54683ff9ab6SJames Feist if (id.key() != "@odata.id") 54783ff9ab6SJames Feist { 54883ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Illegal Type " << id.key(); 54935a62c7cSJason M. Bills messages::propertyUnknown(response->res, field.key()); 55083ff9ab6SJames Feist return CreatePIDRet::fail; 55183ff9ab6SJames Feist } 55283ff9ab6SJames Feist chassisId = id.value().get_ptr<const std::string*>(); 55383ff9ab6SJames Feist if (chassisId == nullptr) 55483ff9ab6SJames Feist { 55583ff9ab6SJames Feist messages::createFailedMissingReqProperties( 55635a62c7cSJason M. Bills response->res, field.key()); 55783ff9ab6SJames Feist return CreatePIDRet::fail; 55883ff9ab6SJames Feist } 55983ff9ab6SJames Feist } 56083ff9ab6SJames Feist 56183ff9ab6SJames Feist // /refish/v1/chassis/chassis_name/ 56283ff9ab6SJames Feist if (!dbus::utility::getNthStringFromPath(*chassisId, 3, 56383ff9ab6SJames Feist chassis)) 56483ff9ab6SJames Feist { 56583ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Got invalid path " << *chassisId; 56635a62c7cSJason M. Bills messages::invalidObject(response->res, *chassisId); 56783ff9ab6SJames Feist return CreatePIDRet::fail; 56883ff9ab6SJames Feist } 56983ff9ab6SJames Feist } 57083ff9ab6SJames Feist else if (field.key() == "FailSafePercent" || 57183ff9ab6SJames Feist field.key() == "MinThermalRpm") 57283ff9ab6SJames Feist { 57383ff9ab6SJames Feist const double* ptr = field.value().get_ptr<const double*>(); 57483ff9ab6SJames Feist if (ptr == nullptr) 57583ff9ab6SJames Feist { 57683ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Illegal Type " << field.key(); 57735a62c7cSJason M. Bills messages::propertyValueFormatError( 57835a62c7cSJason M. Bills response->res, field.value().dump(), field.key()); 57983ff9ab6SJames Feist return CreatePIDRet::fail; 58083ff9ab6SJames Feist } 58183ff9ab6SJames Feist output[field.key()] = *ptr; 58283ff9ab6SJames Feist } 58383ff9ab6SJames Feist else 58483ff9ab6SJames Feist { 58583ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Illegal Type " << field.key(); 58635a62c7cSJason M. Bills messages::propertyUnknown(response->res, field.key()); 58783ff9ab6SJames Feist return CreatePIDRet::fail; 58883ff9ab6SJames Feist } 58983ff9ab6SJames Feist } 59083ff9ab6SJames Feist } 59183ff9ab6SJames Feist else 59283ff9ab6SJames Feist { 59383ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Illegal Type " << type; 59435a62c7cSJason M. Bills messages::propertyUnknown(response->res, type); 59583ff9ab6SJames Feist return CreatePIDRet::fail; 59683ff9ab6SJames Feist } 59783ff9ab6SJames Feist return CreatePIDRet::patch; 59883ff9ab6SJames Feist } 59983ff9ab6SJames Feist 6001abe55efSEd Tanous class Manager : public Node 6011abe55efSEd Tanous { 6029c310685SBorawski.Lukasz public: 6035b4aa86bSJames Feist Manager(CrowApp& app) : Node(app, "/redfish/v1/Managers/bmc/") 6041abe55efSEd Tanous { 605*0f74e643SEd Tanous uuid = app.template getMiddleware<crow::persistent_data::Middleware>() 60655c7b7a2SEd Tanous .systemUuid; 607a434f2bdSEd Tanous entityPrivileges = { 608a434f2bdSEd Tanous {boost::beast::http::verb::get, {{"Login"}}}, 609e0d918bcSEd Tanous {boost::beast::http::verb::head, {{"Login"}}}, 610e0d918bcSEd Tanous {boost::beast::http::verb::patch, {{"ConfigureManager"}}}, 611e0d918bcSEd Tanous {boost::beast::http::verb::put, {{"ConfigureManager"}}}, 612e0d918bcSEd Tanous {boost::beast::http::verb::delete_, {{"ConfigureManager"}}}, 613e0d918bcSEd Tanous {boost::beast::http::verb::post, {{"ConfigureManager"}}}}; 6149c310685SBorawski.Lukasz } 6159c310685SBorawski.Lukasz 6169c310685SBorawski.Lukasz private: 6175b4aa86bSJames Feist void getPidValues(std::shared_ptr<AsyncResp> asyncResp) 6185b4aa86bSJames Feist { 6195b4aa86bSJames Feist crow::connections::systemBus->async_method_call( 6205b4aa86bSJames Feist [asyncResp](const boost::system::error_code ec, 6215b4aa86bSJames Feist const crow::openbmc_mapper::GetSubTreeType& subtree) { 6225b4aa86bSJames Feist if (ec) 6235b4aa86bSJames Feist { 6245b4aa86bSJames Feist BMCWEB_LOG_ERROR << ec; 625f12894f8SJason M. Bills messages::internalError(asyncResp->res); 6265b4aa86bSJames Feist return; 6275b4aa86bSJames Feist } 6285b4aa86bSJames Feist 6295b4aa86bSJames Feist // create map of <connection, path to objMgr>> 6305b4aa86bSJames Feist boost::container::flat_map<std::string, std::string> 6315b4aa86bSJames Feist objectMgrPaths; 6326bce33bcSJames Feist boost::container::flat_set<std::string> calledConnections; 6335b4aa86bSJames Feist for (const auto& pathGroup : subtree) 6345b4aa86bSJames Feist { 6355b4aa86bSJames Feist for (const auto& connectionGroup : pathGroup.second) 6365b4aa86bSJames Feist { 6376bce33bcSJames Feist auto findConnection = 6386bce33bcSJames Feist calledConnections.find(connectionGroup.first); 6396bce33bcSJames Feist if (findConnection != calledConnections.end()) 6406bce33bcSJames Feist { 6416bce33bcSJames Feist break; 6426bce33bcSJames Feist } 6435b4aa86bSJames Feist for (const std::string& interface : 6445b4aa86bSJames Feist connectionGroup.second) 6455b4aa86bSJames Feist { 6465b4aa86bSJames Feist if (interface == objectManagerIface) 6475b4aa86bSJames Feist { 6485b4aa86bSJames Feist objectMgrPaths[connectionGroup.first] = 6495b4aa86bSJames Feist pathGroup.first; 6505b4aa86bSJames Feist } 6515b4aa86bSJames Feist // this list is alphabetical, so we 6525b4aa86bSJames Feist // should have found the objMgr by now 6535b4aa86bSJames Feist if (interface == pidConfigurationIface || 6545b4aa86bSJames Feist interface == pidZoneConfigurationIface) 6555b4aa86bSJames Feist { 6565b4aa86bSJames Feist auto findObjMgr = 6575b4aa86bSJames Feist objectMgrPaths.find(connectionGroup.first); 6585b4aa86bSJames Feist if (findObjMgr == objectMgrPaths.end()) 6595b4aa86bSJames Feist { 6605b4aa86bSJames Feist BMCWEB_LOG_DEBUG << connectionGroup.first 6615b4aa86bSJames Feist << "Has no Object Manager"; 6625b4aa86bSJames Feist continue; 6635b4aa86bSJames Feist } 6646bce33bcSJames Feist 6656bce33bcSJames Feist calledConnections.insert(connectionGroup.first); 6666bce33bcSJames Feist 6675b4aa86bSJames Feist asyncPopulatePid(findObjMgr->first, 6685b4aa86bSJames Feist findObjMgr->second, asyncResp); 6695b4aa86bSJames Feist break; 6705b4aa86bSJames Feist } 6715b4aa86bSJames Feist } 6725b4aa86bSJames Feist } 6735b4aa86bSJames Feist } 6745b4aa86bSJames Feist }, 6755b4aa86bSJames Feist "xyz.openbmc_project.ObjectMapper", 6765b4aa86bSJames Feist "/xyz/openbmc_project/object_mapper", 6775b4aa86bSJames Feist "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0, 6785b4aa86bSJames Feist std::array<const char*, 3>{pidConfigurationIface, 6795b4aa86bSJames Feist pidZoneConfigurationIface, 6805b4aa86bSJames Feist objectManagerIface}); 6815b4aa86bSJames Feist } 6825b4aa86bSJames Feist 68355c7b7a2SEd Tanous void doGet(crow::Response& res, const crow::Request& req, 6841abe55efSEd Tanous const std::vector<std::string>& params) override 6851abe55efSEd Tanous { 686*0f74e643SEd Tanous res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc"; 687*0f74e643SEd Tanous res.jsonValue["@odata.type"] = "#Manager.v1_3_0.Manager"; 688*0f74e643SEd Tanous res.jsonValue["@odata.context"] = 689*0f74e643SEd Tanous "/redfish/v1/$metadata#Manager.Manager"; 690*0f74e643SEd Tanous res.jsonValue["Id"] = "bmc"; 691*0f74e643SEd Tanous res.jsonValue["Name"] = "OpenBmc Manager"; 692*0f74e643SEd Tanous res.jsonValue["Description"] = "Baseboard Management Controller"; 693*0f74e643SEd Tanous res.jsonValue["PowerState"] = "On"; 694*0f74e643SEd Tanous res.jsonValue["ManagerType"] = "BMC"; 695*0f74e643SEd Tanous res.jsonValue["UUID"] = 696*0f74e643SEd Tanous 697*0f74e643SEd Tanous res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model 698*0f74e643SEd Tanous 699*0f74e643SEd Tanous res.jsonValue["LogServices"] = { 700*0f74e643SEd Tanous {"@odata.id", "/redfish/v1/Managers/bmc/LogServices"}}; 701*0f74e643SEd Tanous 702*0f74e643SEd Tanous res.jsonValue["NetworkProtocol"] = { 703*0f74e643SEd Tanous {"@odata.id", "/redfish/v1/Managers/bmc/NetworkProtocol"}}; 704*0f74e643SEd Tanous 705*0f74e643SEd Tanous res.jsonValue["EthernetInterfaces"] = { 706*0f74e643SEd Tanous {"@odata.id", "/redfish/v1/Managers/bmc/EthernetInterfaces"}}; 707*0f74e643SEd Tanous // default oem data 708*0f74e643SEd Tanous nlohmann::json& oem = res.jsonValue["Oem"]; 709*0f74e643SEd Tanous nlohmann::json& oemOpenbmc = oem["OpenBmc"]; 710*0f74e643SEd Tanous oem["@odata.type"] = "#OemManager.Oem"; 711*0f74e643SEd Tanous oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem"; 712*0f74e643SEd Tanous oem["@odata.context"] = "/redfish/v1/$metadata#OemManager.Oem"; 713*0f74e643SEd Tanous oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc"; 714*0f74e643SEd Tanous oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc"; 715*0f74e643SEd Tanous oemOpenbmc["@odata.context"] = 716*0f74e643SEd Tanous "/redfish/v1/$metadata#OemManager.OpenBmc"; 717*0f74e643SEd Tanous 718ed5befbdSJennifer Lee // Update Actions object. 719*0f74e643SEd Tanous nlohmann::json& manager_reset = 720*0f74e643SEd Tanous res.jsonValue["Actions"]["#Manager.Reset"]; 721ed5befbdSJennifer Lee manager_reset["target"] = 722ed5befbdSJennifer Lee "/redfish/v1/Managers/bmc/Actions/Manager.Reset"; 723ed5befbdSJennifer Lee manager_reset["ResetType@Redfish.AllowableValues"] = { 724ed5befbdSJennifer Lee "GracefulRestart"}; 725ca537928SJennifer Lee 726*0f74e643SEd Tanous res.jsonValue["DateTime"] = getDateTime(); 727ed5befbdSJennifer Lee std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res); 7285b4aa86bSJames Feist 729ca537928SJennifer Lee crow::connections::systemBus->async_method_call( 730ca537928SJennifer Lee [asyncResp](const boost::system::error_code ec, 7315b4aa86bSJames Feist const dbus::utility::ManagedObjectType& resp) { 732ca537928SJennifer Lee if (ec) 733ca537928SJennifer Lee { 734ca537928SJennifer Lee BMCWEB_LOG_ERROR << "Error while getting Software Version"; 735f12894f8SJason M. Bills messages::internalError(asyncResp->res); 736ca537928SJennifer Lee return; 737ca537928SJennifer Lee } 738ca537928SJennifer Lee 739ca537928SJennifer Lee for (auto& objpath : resp) 740ca537928SJennifer Lee { 741ca537928SJennifer Lee for (auto& interface : objpath.second) 742ca537928SJennifer Lee { 743ca537928SJennifer Lee // If interface is xyz.openbmc_project.Software.Version, 744ca537928SJennifer Lee // this is what we're looking for. 745ca537928SJennifer Lee if (interface.first == 746ca537928SJennifer Lee "xyz.openbmc_project.Software.Version") 747ca537928SJennifer Lee { 748ca537928SJennifer Lee // Cut out everyting until last "/", ... 749ca537928SJennifer Lee const std::string& iface_id = objpath.first; 750ca537928SJennifer Lee for (auto& property : interface.second) 751ca537928SJennifer Lee { 752ca537928SJennifer Lee if (property.first == "Version") 753ca537928SJennifer Lee { 754ca537928SJennifer Lee const std::string* value = 755ca537928SJennifer Lee mapbox::getPtr<const std::string>( 756ca537928SJennifer Lee property.second); 757ca537928SJennifer Lee if (value == nullptr) 758ca537928SJennifer Lee { 759ca537928SJennifer Lee continue; 760ca537928SJennifer Lee } 761ca537928SJennifer Lee asyncResp->res 762ca537928SJennifer Lee .jsonValue["FirmwareVersion"] = *value; 763ca537928SJennifer Lee } 764ca537928SJennifer Lee } 765ca537928SJennifer Lee } 766ca537928SJennifer Lee } 767ca537928SJennifer Lee } 768ca537928SJennifer Lee }, 769ca537928SJennifer Lee "xyz.openbmc_project.Software.BMC.Updater", 770ca537928SJennifer Lee "/xyz/openbmc_project/software", 771ca537928SJennifer Lee "org.freedesktop.DBus.ObjectManager", "GetManagedObjects"); 7725b4aa86bSJames Feist getPidValues(asyncResp); 7735b4aa86bSJames Feist } 77483ff9ab6SJames Feist void setPidValues(std::shared_ptr<AsyncResp> response, 77583ff9ab6SJames Feist const nlohmann::json& data) 77683ff9ab6SJames Feist { 77783ff9ab6SJames Feist // todo(james): might make sense to do a mapper call here if this 77883ff9ab6SJames Feist // interface gets more traction 77983ff9ab6SJames Feist crow::connections::systemBus->async_method_call( 78083ff9ab6SJames Feist [response, 78183ff9ab6SJames Feist data](const boost::system::error_code ec, 78283ff9ab6SJames Feist const dbus::utility::ManagedObjectType& managedObj) { 78383ff9ab6SJames Feist if (ec) 78483ff9ab6SJames Feist { 78583ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Error communicating to Entity Manager"; 78635a62c7cSJason M. Bills messages::internalError(response->res); 78783ff9ab6SJames Feist return; 78883ff9ab6SJames Feist } 78983ff9ab6SJames Feist for (const auto& type : data.items()) 79083ff9ab6SJames Feist { 79183ff9ab6SJames Feist if (!type.value().is_object()) 79283ff9ab6SJames Feist { 79383ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Illegal Type " << type.key(); 79435a62c7cSJason M. Bills messages::propertyValueFormatError( 79535a62c7cSJason M. Bills response->res, type.value(), type.key()); 79683ff9ab6SJames Feist return; 79783ff9ab6SJames Feist } 79883ff9ab6SJames Feist for (const auto& record : type.value().items()) 79983ff9ab6SJames Feist { 80083ff9ab6SJames Feist const std::string& name = record.key(); 80183ff9ab6SJames Feist auto pathItr = 80283ff9ab6SJames Feist std::find_if(managedObj.begin(), managedObj.end(), 80383ff9ab6SJames Feist [&name](const auto& obj) { 80483ff9ab6SJames Feist return boost::algorithm::ends_with( 80583ff9ab6SJames Feist obj.first.str, name); 80683ff9ab6SJames Feist }); 80783ff9ab6SJames Feist boost::container::flat_map< 80883ff9ab6SJames Feist std::string, dbus::utility::DbusVariantType> 80983ff9ab6SJames Feist output; 81083ff9ab6SJames Feist 81183ff9ab6SJames Feist output.reserve(16); // The pid interface length 81283ff9ab6SJames Feist 81383ff9ab6SJames Feist // determines if we're patching entity-manager or 81483ff9ab6SJames Feist // creating a new object 81583ff9ab6SJames Feist bool createNewObject = (pathItr == managedObj.end()); 81683ff9ab6SJames Feist if (type.key() == "PidControllers" || 81783ff9ab6SJames Feist type.key() == "FanControllers") 81883ff9ab6SJames Feist { 81983ff9ab6SJames Feist if (!createNewObject && 82083ff9ab6SJames Feist pathItr->second.find(pidConfigurationIface) == 82183ff9ab6SJames Feist pathItr->second.end()) 82283ff9ab6SJames Feist { 82383ff9ab6SJames Feist createNewObject = true; 82483ff9ab6SJames Feist } 82583ff9ab6SJames Feist } 82683ff9ab6SJames Feist else if (!createNewObject && 82783ff9ab6SJames Feist pathItr->second.find( 82883ff9ab6SJames Feist pidZoneConfigurationIface) == 82983ff9ab6SJames Feist pathItr->second.end()) 83083ff9ab6SJames Feist { 83183ff9ab6SJames Feist createNewObject = true; 83283ff9ab6SJames Feist } 83383ff9ab6SJames Feist output["Name"] = 83483ff9ab6SJames Feist boost::replace_all_copy(name, "_", " "); 83583ff9ab6SJames Feist 83683ff9ab6SJames Feist std::string chassis; 83783ff9ab6SJames Feist CreatePIDRet ret = createPidInterface( 83883ff9ab6SJames Feist response, type.key(), record.value(), 83983ff9ab6SJames Feist pathItr->first.str, managedObj, createNewObject, 84083ff9ab6SJames Feist output, chassis); 84183ff9ab6SJames Feist if (ret == CreatePIDRet::fail) 84283ff9ab6SJames Feist { 84383ff9ab6SJames Feist return; 84483ff9ab6SJames Feist } 84583ff9ab6SJames Feist else if (ret == CreatePIDRet::del) 84683ff9ab6SJames Feist { 84783ff9ab6SJames Feist continue; 84883ff9ab6SJames Feist } 84983ff9ab6SJames Feist 85083ff9ab6SJames Feist if (!createNewObject) 85183ff9ab6SJames Feist { 85283ff9ab6SJames Feist for (const auto& property : output) 85383ff9ab6SJames Feist { 85483ff9ab6SJames Feist const char* iface = 85583ff9ab6SJames Feist type.key() == "FanZones" 85683ff9ab6SJames Feist ? pidZoneConfigurationIface 85783ff9ab6SJames Feist : pidConfigurationIface; 85883ff9ab6SJames Feist crow::connections::systemBus->async_method_call( 85983ff9ab6SJames Feist [response, 86083ff9ab6SJames Feist propertyName{std::string(property.first)}]( 86183ff9ab6SJames Feist const boost::system::error_code ec) { 86283ff9ab6SJames Feist if (ec) 86383ff9ab6SJames Feist { 86483ff9ab6SJames Feist BMCWEB_LOG_ERROR 86583ff9ab6SJames Feist << "Error patching " 86683ff9ab6SJames Feist << propertyName << ": " << ec; 86735a62c7cSJason M. Bills messages::internalError( 86835a62c7cSJason M. Bills response->res); 86983ff9ab6SJames Feist } 87083ff9ab6SJames Feist }, 87183ff9ab6SJames Feist "xyz.openbmc_project.EntityManager", 87283ff9ab6SJames Feist pathItr->first.str, 87383ff9ab6SJames Feist "org.freedesktop.DBus.Properties", "Set", 87483ff9ab6SJames Feist std::string(iface), property.first, 87583ff9ab6SJames Feist property.second); 87683ff9ab6SJames Feist } 87783ff9ab6SJames Feist } 87883ff9ab6SJames Feist else 87983ff9ab6SJames Feist { 88083ff9ab6SJames Feist if (chassis.empty()) 88183ff9ab6SJames Feist { 88283ff9ab6SJames Feist BMCWEB_LOG_ERROR 88383ff9ab6SJames Feist << "Failed to get chassis from config"; 88435a62c7cSJason M. Bills messages::invalidObject(response->res, name); 88583ff9ab6SJames Feist return; 88683ff9ab6SJames Feist } 88783ff9ab6SJames Feist 88883ff9ab6SJames Feist bool foundChassis = false; 88983ff9ab6SJames Feist for (const auto& obj : managedObj) 89083ff9ab6SJames Feist { 89183ff9ab6SJames Feist if (boost::algorithm::ends_with(obj.first.str, 89283ff9ab6SJames Feist chassis)) 89383ff9ab6SJames Feist { 89483ff9ab6SJames Feist chassis = obj.first.str; 89583ff9ab6SJames Feist foundChassis = true; 89683ff9ab6SJames Feist break; 89783ff9ab6SJames Feist } 89883ff9ab6SJames Feist } 89983ff9ab6SJames Feist if (!foundChassis) 90083ff9ab6SJames Feist { 90183ff9ab6SJames Feist BMCWEB_LOG_ERROR 90283ff9ab6SJames Feist << "Failed to find chassis on dbus"; 90383ff9ab6SJames Feist messages::resourceMissingAtURI( 90435a62c7cSJason M. Bills response->res, 90535a62c7cSJason M. Bills "/redfish/v1/Chassis/" + chassis); 90683ff9ab6SJames Feist return; 90783ff9ab6SJames Feist } 90883ff9ab6SJames Feist 90983ff9ab6SJames Feist crow::connections::systemBus->async_method_call( 91083ff9ab6SJames Feist [response](const boost::system::error_code ec) { 91183ff9ab6SJames Feist if (ec) 91283ff9ab6SJames Feist { 91383ff9ab6SJames Feist BMCWEB_LOG_ERROR 91483ff9ab6SJames Feist << "Error Adding Pid Object " << ec; 91535a62c7cSJason M. Bills messages::internalError(response->res); 91683ff9ab6SJames Feist } 91783ff9ab6SJames Feist }, 91883ff9ab6SJames Feist "xyz.openbmc_project.EntityManager", chassis, 91983ff9ab6SJames Feist "xyz.openbmc_project.AddObject", "AddObject", 92083ff9ab6SJames Feist output); 92183ff9ab6SJames Feist } 92283ff9ab6SJames Feist } 92383ff9ab6SJames Feist } 92483ff9ab6SJames Feist }, 92583ff9ab6SJames Feist "xyz.openbmc_project.EntityManager", "/", objectManagerIface, 92683ff9ab6SJames Feist "GetManagedObjects"); 92783ff9ab6SJames Feist } 9285b4aa86bSJames Feist 9295b4aa86bSJames Feist void doPatch(crow::Response& res, const crow::Request& req, 9305b4aa86bSJames Feist const std::vector<std::string>& params) override 9315b4aa86bSJames Feist { 93283ff9ab6SJames Feist nlohmann::json patch; 93383ff9ab6SJames Feist if (!json_util::processJsonFromRequest(res, req, patch)) 93483ff9ab6SJames Feist { 93583ff9ab6SJames Feist return; 93683ff9ab6SJames Feist } 93783ff9ab6SJames Feist std::shared_ptr<AsyncResp> response = std::make_shared<AsyncResp>(res); 93883ff9ab6SJames Feist for (const auto& topLevel : patch.items()) 93983ff9ab6SJames Feist { 94083ff9ab6SJames Feist if (topLevel.key() == "Oem") 94183ff9ab6SJames Feist { 94283ff9ab6SJames Feist if (!topLevel.value().is_object()) 94383ff9ab6SJames Feist { 94483ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Bad Patch " << topLevel.key(); 94535a62c7cSJason M. Bills messages::propertyValueFormatError( 94635a62c7cSJason M. Bills response->res, topLevel.key(), "OemManager.Oem"); 94783ff9ab6SJames Feist return; 94883ff9ab6SJames Feist } 94983ff9ab6SJames Feist } 95083ff9ab6SJames Feist else 95183ff9ab6SJames Feist { 95283ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Bad Patch " << topLevel.key(); 95335a62c7cSJason M. Bills messages::propertyUnknown(response->res, topLevel.key()); 95483ff9ab6SJames Feist return; 95583ff9ab6SJames Feist } 95683ff9ab6SJames Feist for (const auto& oemLevel : topLevel.value().items()) 95783ff9ab6SJames Feist { 95883ff9ab6SJames Feist if (oemLevel.key() == "OpenBmc") 95983ff9ab6SJames Feist { 96083ff9ab6SJames Feist if (!oemLevel.value().is_object()) 96183ff9ab6SJames Feist { 96283ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Bad Patch " << oemLevel.key(); 96335a62c7cSJason M. Bills messages::propertyValueFormatError( 96435a62c7cSJason M. Bills response->res, topLevel.key(), 96535a62c7cSJason M. Bills "OemManager.OpenBmc"); 96683ff9ab6SJames Feist return; 96783ff9ab6SJames Feist } 96883ff9ab6SJames Feist for (const auto& typeLevel : oemLevel.value().items()) 96983ff9ab6SJames Feist { 97083ff9ab6SJames Feist 97183ff9ab6SJames Feist if (typeLevel.key() == "Fan") 97283ff9ab6SJames Feist { 97383ff9ab6SJames Feist if (!typeLevel.value().is_object()) 97483ff9ab6SJames Feist { 97583ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Bad Patch " 97683ff9ab6SJames Feist << typeLevel.key(); 97783ff9ab6SJames Feist messages::propertyValueFormatError( 97835a62c7cSJason M. Bills response->res, typeLevel.value().dump(), 97935a62c7cSJason M. Bills typeLevel.key()); 98083ff9ab6SJames Feist return; 98183ff9ab6SJames Feist } 98283ff9ab6SJames Feist setPidValues(response, 98383ff9ab6SJames Feist std::move(typeLevel.value())); 98483ff9ab6SJames Feist } 98583ff9ab6SJames Feist else 98683ff9ab6SJames Feist { 98783ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Bad Patch " << typeLevel.key(); 98835a62c7cSJason M. Bills messages::propertyUnknown(response->res, 98935a62c7cSJason M. Bills typeLevel.key()); 99083ff9ab6SJames Feist return; 99183ff9ab6SJames Feist } 99283ff9ab6SJames Feist } 99383ff9ab6SJames Feist } 99483ff9ab6SJames Feist else 99583ff9ab6SJames Feist { 99683ff9ab6SJames Feist BMCWEB_LOG_ERROR << "Bad Patch " << oemLevel.key(); 99735a62c7cSJason M. Bills messages::propertyUnknown(response->res, oemLevel.key()); 99883ff9ab6SJames Feist return; 99983ff9ab6SJames Feist } 100083ff9ab6SJames Feist } 100183ff9ab6SJames Feist } 10029c310685SBorawski.Lukasz } 10039c310685SBorawski.Lukasz 10041abe55efSEd Tanous std::string getDateTime() const 10051abe55efSEd Tanous { 10069c310685SBorawski.Lukasz std::array<char, 128> dateTime; 10079c310685SBorawski.Lukasz std::string redfishDateTime("0000-00-00T00:00:00Z00:00"); 10089c310685SBorawski.Lukasz std::time_t time = std::time(nullptr); 10099c310685SBorawski.Lukasz 10109c310685SBorawski.Lukasz if (std::strftime(dateTime.begin(), dateTime.size(), "%FT%T%z", 10111abe55efSEd Tanous std::localtime(&time))) 10121abe55efSEd Tanous { 10139c310685SBorawski.Lukasz // insert the colon required by the ISO 8601 standard 10149c310685SBorawski.Lukasz redfishDateTime = std::string(dateTime.data()); 10159c310685SBorawski.Lukasz redfishDateTime.insert(redfishDateTime.end() - 2, ':'); 10169c310685SBorawski.Lukasz } 10179c310685SBorawski.Lukasz 10189c310685SBorawski.Lukasz return redfishDateTime; 10199c310685SBorawski.Lukasz } 1020*0f74e643SEd Tanous 1021*0f74e643SEd Tanous std::string uuid; 10229c310685SBorawski.Lukasz }; 10239c310685SBorawski.Lukasz 10241abe55efSEd Tanous class ManagerCollection : public Node 10251abe55efSEd Tanous { 10269c310685SBorawski.Lukasz public: 10271abe55efSEd Tanous ManagerCollection(CrowApp& app) : Node(app, "/redfish/v1/Managers/") 10281abe55efSEd Tanous { 1029a434f2bdSEd Tanous entityPrivileges = { 1030a434f2bdSEd Tanous {boost::beast::http::verb::get, {{"Login"}}}, 1031e0d918bcSEd Tanous {boost::beast::http::verb::head, {{"Login"}}}, 1032e0d918bcSEd Tanous {boost::beast::http::verb::patch, {{"ConfigureManager"}}}, 1033e0d918bcSEd Tanous {boost::beast::http::verb::put, {{"ConfigureManager"}}}, 1034e0d918bcSEd Tanous {boost::beast::http::verb::delete_, {{"ConfigureManager"}}}, 1035e0d918bcSEd Tanous {boost::beast::http::verb::post, {{"ConfigureManager"}}}}; 10369c310685SBorawski.Lukasz } 10379c310685SBorawski.Lukasz 10389c310685SBorawski.Lukasz private: 103955c7b7a2SEd Tanous void doGet(crow::Response& res, const crow::Request& req, 10401abe55efSEd Tanous const std::vector<std::string>& params) override 10411abe55efSEd Tanous { 104283ff9ab6SJames Feist // Collections don't include the static data added by SubRoute 104383ff9ab6SJames Feist // because it has a duplicate entry for members 104455c7b7a2SEd Tanous res.jsonValue["@odata.id"] = "/redfish/v1/Managers"; 104555c7b7a2SEd Tanous res.jsonValue["@odata.type"] = "#ManagerCollection.ManagerCollection"; 104655c7b7a2SEd Tanous res.jsonValue["@odata.context"] = 104755c7b7a2SEd Tanous "/redfish/v1/$metadata#ManagerCollection.ManagerCollection"; 104855c7b7a2SEd Tanous res.jsonValue["Name"] = "Manager Collection"; 104955c7b7a2SEd Tanous res.jsonValue["Members@odata.count"] = 1; 105055c7b7a2SEd Tanous res.jsonValue["Members"] = { 10515b4aa86bSJames Feist {{"@odata.id", "/redfish/v1/Managers/bmc"}}}; 10529c310685SBorawski.Lukasz res.end(); 10539c310685SBorawski.Lukasz } 10549c310685SBorawski.Lukasz }; 10559c310685SBorawski.Lukasz } // namespace redfish 1056