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 1813451e39SWilly Tu #include "bmcweb_config.h" 1913451e39SWilly Tu 20a51fc2d2SSui Chen #include "app.hpp" 21a51fc2d2SSui Chen #include "dbus_utility.hpp" 22b49ac873SJames Feist #include "health.hpp" 23a51fc2d2SSui Chen #include "query.hpp" 24c5d03ff4SJennifer Lee #include "redfish_util.hpp" 25a51fc2d2SSui Chen #include "registries/privilege_registry.hpp" 26fac6e53bSKrzysztof Grobelny #include "utils/dbus_utils.hpp" 273ccb3adbSEd Tanous #include "utils/json_utils.hpp" 28a51fc2d2SSui Chen #include "utils/sw_utils.hpp" 29a51fc2d2SSui Chen #include "utils/systemd_utils.hpp" 302b82937eSEd Tanous #include "utils/time_utils.hpp" 319c310685SBorawski.Lukasz 32e99073f5SGeorge Liu #include <boost/system/error_code.hpp> 33ef4c65b7SEd Tanous #include <boost/url/format.hpp> 34fac6e53bSKrzysztof Grobelny #include <sdbusplus/asio/property.hpp> 35fac6e53bSKrzysztof Grobelny #include <sdbusplus/unpack_properties.hpp> 361214b7e7SGunnar Mills 37a170f275SEd Tanous #include <algorithm> 38e99073f5SGeorge Liu #include <array> 394bfefa74SGunnar Mills #include <cstdint> 401214b7e7SGunnar Mills #include <memory> 419970e93fSKonstantin Aladyshev #include <optional> 423544d2a7SEd Tanous #include <ranges> 431214b7e7SGunnar Mills #include <sstream> 449970e93fSKonstantin Aladyshev #include <string> 45e99073f5SGeorge Liu #include <string_view> 46abf2add6SEd Tanous #include <variant> 475b4aa86bSJames Feist 481abe55efSEd Tanous namespace redfish 491abe55efSEd Tanous { 50ed5befbdSJennifer Lee 51ed5befbdSJennifer Lee /** 522a5c4407SGunnar Mills * Function reboots the BMC. 532a5c4407SGunnar Mills * 542a5c4407SGunnar Mills * @param[in] asyncResp - Shared pointer for completing asynchronous calls 55ed5befbdSJennifer Lee */ 568d1b46d7Szhanghch05 inline void 578d1b46d7Szhanghch05 doBMCGracefulRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 58ed5befbdSJennifer Lee { 59ed5befbdSJennifer Lee const char* processName = "xyz.openbmc_project.State.BMC"; 60ed5befbdSJennifer Lee const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 61ed5befbdSJennifer Lee const char* interfaceName = "xyz.openbmc_project.State.BMC"; 62ed5befbdSJennifer Lee const std::string& propertyValue = 63ed5befbdSJennifer Lee "xyz.openbmc_project.State.BMC.Transition.Reboot"; 64ed5befbdSJennifer Lee const char* destProperty = "RequestedBMCTransition"; 65ed5befbdSJennifer Lee 66ed5befbdSJennifer Lee // Create the D-Bus variant for D-Bus call. 679ae226faSGeorge Liu sdbusplus::asio::setProperty( 689ae226faSGeorge Liu *crow::connections::systemBus, processName, objectPath, interfaceName, 699ae226faSGeorge Liu destProperty, propertyValue, 705e7e2dc5SEd Tanous [asyncResp](const boost::system::error_code& ec) { 71ed5befbdSJennifer Lee // Use "Set" method to set the property value. 72ed5befbdSJennifer Lee if (ec) 73ed5befbdSJennifer Lee { 7462598e31SEd Tanous BMCWEB_LOG_DEBUG("[Set] Bad D-Bus request error: {}", ec); 75ed5befbdSJennifer Lee messages::internalError(asyncResp->res); 76ed5befbdSJennifer Lee return; 77ed5befbdSJennifer Lee } 78ed5befbdSJennifer Lee 79ed5befbdSJennifer Lee messages::success(asyncResp->res); 809ae226faSGeorge Liu }); 81ed5befbdSJennifer Lee } 822a5c4407SGunnar Mills 838d1b46d7Szhanghch05 inline void 848d1b46d7Szhanghch05 doBMCForceRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 85f92af389SJayaprakash Mutyala { 86f92af389SJayaprakash Mutyala const char* processName = "xyz.openbmc_project.State.BMC"; 87f92af389SJayaprakash Mutyala const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 88f92af389SJayaprakash Mutyala const char* interfaceName = "xyz.openbmc_project.State.BMC"; 89f92af389SJayaprakash Mutyala const std::string& propertyValue = 90f92af389SJayaprakash Mutyala "xyz.openbmc_project.State.BMC.Transition.HardReboot"; 91f92af389SJayaprakash Mutyala const char* destProperty = "RequestedBMCTransition"; 92f92af389SJayaprakash Mutyala 93f92af389SJayaprakash Mutyala // Create the D-Bus variant for D-Bus call. 949ae226faSGeorge Liu sdbusplus::asio::setProperty( 959ae226faSGeorge Liu *crow::connections::systemBus, processName, objectPath, interfaceName, 969ae226faSGeorge Liu destProperty, propertyValue, 975e7e2dc5SEd Tanous [asyncResp](const boost::system::error_code& ec) { 98f92af389SJayaprakash Mutyala // Use "Set" method to set the property value. 99f92af389SJayaprakash Mutyala if (ec) 100f92af389SJayaprakash Mutyala { 10162598e31SEd Tanous BMCWEB_LOG_DEBUG("[Set] Bad D-Bus request error: {}", ec); 102f92af389SJayaprakash Mutyala messages::internalError(asyncResp->res); 103f92af389SJayaprakash Mutyala return; 104f92af389SJayaprakash Mutyala } 105f92af389SJayaprakash Mutyala 106f92af389SJayaprakash Mutyala messages::success(asyncResp->res); 1079ae226faSGeorge Liu }); 108f92af389SJayaprakash Mutyala } 109f92af389SJayaprakash Mutyala 1102a5c4407SGunnar Mills /** 1112a5c4407SGunnar Mills * ManagerResetAction class supports the POST method for the Reset (reboot) 1122a5c4407SGunnar Mills * action. 1132a5c4407SGunnar Mills */ 1147e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetAction(App& app) 1152a5c4407SGunnar Mills { 1162a5c4407SGunnar Mills /** 1172a5c4407SGunnar Mills * Function handles POST method request. 1182a5c4407SGunnar Mills * Analyzes POST body before sending Reset (Reboot) request data to D-Bus. 119f92af389SJayaprakash Mutyala * OpenBMC supports ResetType "GracefulRestart" and "ForceRestart". 1202a5c4407SGunnar Mills */ 1217e860f15SJohn Edward Broadbent 1227e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/") 123ed398213SEd Tanous .privileges(redfish::privileges::postManager) 1247e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::post)( 12545ca1b86SEd Tanous [&app](const crow::Request& req, 1267e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 1273ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 12845ca1b86SEd Tanous { 12945ca1b86SEd Tanous return; 13045ca1b86SEd Tanous } 13162598e31SEd Tanous BMCWEB_LOG_DEBUG("Post Manager Reset."); 1322a5c4407SGunnar Mills 1332a5c4407SGunnar Mills std::string resetType; 1342a5c4407SGunnar Mills 13515ed6780SWilly Tu if (!json_util::readJsonAction(req, asyncResp->res, "ResetType", 1367e860f15SJohn Edward Broadbent resetType)) 1372a5c4407SGunnar Mills { 1382a5c4407SGunnar Mills return; 1392a5c4407SGunnar Mills } 1402a5c4407SGunnar Mills 141f92af389SJayaprakash Mutyala if (resetType == "GracefulRestart") 142f92af389SJayaprakash Mutyala { 14362598e31SEd Tanous BMCWEB_LOG_DEBUG("Proceeding with {}", resetType); 144f92af389SJayaprakash Mutyala doBMCGracefulRestart(asyncResp); 145f92af389SJayaprakash Mutyala return; 146f92af389SJayaprakash Mutyala } 1473174e4dfSEd Tanous if (resetType == "ForceRestart") 148f92af389SJayaprakash Mutyala { 14962598e31SEd Tanous BMCWEB_LOG_DEBUG("Proceeding with {}", resetType); 150f92af389SJayaprakash Mutyala doBMCForceRestart(asyncResp); 151f92af389SJayaprakash Mutyala return; 152f92af389SJayaprakash Mutyala } 15362598e31SEd Tanous BMCWEB_LOG_DEBUG("Invalid property value for ResetType: {}", resetType); 1542a5c4407SGunnar Mills messages::actionParameterNotSupported(asyncResp->res, resetType, 1552a5c4407SGunnar Mills "ResetType"); 1562a5c4407SGunnar Mills 1572a5c4407SGunnar Mills return; 1587e860f15SJohn Edward Broadbent }); 1592a5c4407SGunnar Mills } 160ed5befbdSJennifer Lee 1613e40fc74SGunnar Mills /** 1623e40fc74SGunnar Mills * ManagerResetToDefaultsAction class supports POST method for factory reset 1633e40fc74SGunnar Mills * action. 1643e40fc74SGunnar Mills */ 1657e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetToDefaultsAction(App& app) 1663e40fc74SGunnar Mills { 1673e40fc74SGunnar Mills /** 1683e40fc74SGunnar Mills * Function handles ResetToDefaults POST method request. 1693e40fc74SGunnar Mills * 1703e40fc74SGunnar Mills * Analyzes POST body message and factory resets BMC by calling 1713e40fc74SGunnar Mills * BMC code updater factory reset followed by a BMC reboot. 1723e40fc74SGunnar Mills * 1733e40fc74SGunnar Mills * BMC code updater factory reset wipes the whole BMC read-write 1743e40fc74SGunnar Mills * filesystem which includes things like the network settings. 1753e40fc74SGunnar Mills * 1763e40fc74SGunnar Mills * OpenBMC only supports ResetToDefaultsType "ResetAll". 1773e40fc74SGunnar Mills */ 1787e860f15SJohn Edward Broadbent 1797e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, 1807e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults/") 181ed398213SEd Tanous .privileges(redfish::privileges::postManager) 1827e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::post)( 18345ca1b86SEd Tanous [&app](const crow::Request& req, 1847e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 1853ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 18645ca1b86SEd Tanous { 18745ca1b86SEd Tanous return; 18845ca1b86SEd Tanous } 18962598e31SEd Tanous BMCWEB_LOG_DEBUG("Post ResetToDefaults."); 1903e40fc74SGunnar Mills 1919970e93fSKonstantin Aladyshev std::optional<std::string> resetType; 1929970e93fSKonstantin Aladyshev std::optional<std::string> resetToDefaultsType; 1933e40fc74SGunnar Mills 1949970e93fSKonstantin Aladyshev if (!json_util::readJsonAction(req, asyncResp->res, "ResetType", 1959970e93fSKonstantin Aladyshev resetType, "ResetToDefaultsType", 1969970e93fSKonstantin Aladyshev resetToDefaultsType)) 1973e40fc74SGunnar Mills { 1989970e93fSKonstantin Aladyshev BMCWEB_LOG_DEBUG("Missing property ResetType."); 1993e40fc74SGunnar Mills 200002d39b4SEd Tanous messages::actionParameterMissing(asyncResp->res, "ResetToDefaults", 2019970e93fSKonstantin Aladyshev "ResetType"); 2023e40fc74SGunnar Mills return; 2033e40fc74SGunnar Mills } 2043e40fc74SGunnar Mills 2059970e93fSKonstantin Aladyshev if (resetToDefaultsType && !resetType) 2069970e93fSKonstantin Aladyshev { 2079970e93fSKonstantin Aladyshev BMCWEB_LOG_WARNING( 2089970e93fSKonstantin Aladyshev "Using deprecated ResetToDefaultsType, should be ResetType." 2099970e93fSKonstantin Aladyshev "Support for the ResetToDefaultsType will be dropped in 2Q24"); 2109970e93fSKonstantin Aladyshev resetType = resetToDefaultsType; 2119970e93fSKonstantin Aladyshev } 2129970e93fSKonstantin Aladyshev 2133e40fc74SGunnar Mills if (resetType != "ResetAll") 2143e40fc74SGunnar Mills { 2159970e93fSKonstantin Aladyshev BMCWEB_LOG_DEBUG("Invalid property value for ResetType: {}", 2169970e93fSKonstantin Aladyshev *resetType); 2179970e93fSKonstantin Aladyshev messages::actionParameterNotSupported(asyncResp->res, *resetType, 2189970e93fSKonstantin Aladyshev "ResetType"); 2193e40fc74SGunnar Mills return; 2203e40fc74SGunnar Mills } 2213e40fc74SGunnar Mills 2223e40fc74SGunnar Mills crow::connections::systemBus->async_method_call( 2235e7e2dc5SEd Tanous [asyncResp](const boost::system::error_code& ec) { 2243e40fc74SGunnar Mills if (ec) 2253e40fc74SGunnar Mills { 22662598e31SEd Tanous BMCWEB_LOG_DEBUG("Failed to ResetToDefaults: {}", ec); 2273e40fc74SGunnar Mills messages::internalError(asyncResp->res); 2283e40fc74SGunnar Mills return; 2293e40fc74SGunnar Mills } 2303e40fc74SGunnar Mills // Factory Reset doesn't actually happen until a reboot 2313e40fc74SGunnar Mills // Can't erase what the BMC is running on 2323e40fc74SGunnar Mills doBMCGracefulRestart(asyncResp); 2333e40fc74SGunnar Mills }, 2343e40fc74SGunnar Mills "xyz.openbmc_project.Software.BMC.Updater", 2353e40fc74SGunnar Mills "/xyz/openbmc_project/software", 2363e40fc74SGunnar Mills "xyz.openbmc_project.Common.FactoryReset", "Reset"); 2377e860f15SJohn Edward Broadbent }); 2383e40fc74SGunnar Mills } 2393e40fc74SGunnar Mills 2401cb1a9e6SAppaRao Puli /** 2411cb1a9e6SAppaRao Puli * ManagerResetActionInfo derived class for delivering Manager 2421cb1a9e6SAppaRao Puli * ResetType AllowableValues using ResetInfo schema. 2431cb1a9e6SAppaRao Puli */ 2447e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetActionInfo(App& app) 2451cb1a9e6SAppaRao Puli { 2461cb1a9e6SAppaRao Puli /** 2471cb1a9e6SAppaRao Puli * Functions triggers appropriate requests on DBus 2481cb1a9e6SAppaRao Puli */ 2497e860f15SJohn Edward Broadbent 2507e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/ResetActionInfo/") 251ed398213SEd Tanous .privileges(redfish::privileges::getActionInfo) 2527e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)( 25345ca1b86SEd Tanous [&app](const crow::Request& req, 2547e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 2553ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 25645ca1b86SEd Tanous { 25745ca1b86SEd Tanous return; 25845ca1b86SEd Tanous } 2591476687dSEd Tanous 2601476687dSEd Tanous asyncResp->res.jsonValue["@odata.type"] = 2611476687dSEd Tanous "#ActionInfo.v1_1_2.ActionInfo"; 2621476687dSEd Tanous asyncResp->res.jsonValue["@odata.id"] = 2631476687dSEd Tanous "/redfish/v1/Managers/bmc/ResetActionInfo"; 2641476687dSEd Tanous asyncResp->res.jsonValue["Name"] = "Reset Action Info"; 2651476687dSEd Tanous asyncResp->res.jsonValue["Id"] = "ResetActionInfo"; 2661476687dSEd Tanous nlohmann::json::object_t parameter; 2671476687dSEd Tanous parameter["Name"] = "ResetType"; 2681476687dSEd Tanous parameter["Required"] = true; 2691476687dSEd Tanous parameter["DataType"] = "String"; 2701476687dSEd Tanous 2711476687dSEd Tanous nlohmann::json::array_t allowableValues; 272ad539545SPatrick Williams allowableValues.emplace_back("GracefulRestart"); 273ad539545SPatrick Williams allowableValues.emplace_back("ForceRestart"); 2741476687dSEd Tanous parameter["AllowableValues"] = std::move(allowableValues); 2751476687dSEd Tanous 2761476687dSEd Tanous nlohmann::json::array_t parameters; 277ad539545SPatrick Williams parameters.emplace_back(std::move(parameter)); 2781476687dSEd Tanous 2791476687dSEd Tanous asyncResp->res.jsonValue["Parameters"] = std::move(parameters); 2807e860f15SJohn Edward Broadbent }); 2811cb1a9e6SAppaRao Puli } 2821cb1a9e6SAppaRao Puli 2835b4aa86bSJames Feist static constexpr const char* objectManagerIface = 2845b4aa86bSJames Feist "org.freedesktop.DBus.ObjectManager"; 2855b4aa86bSJames Feist static constexpr const char* pidConfigurationIface = 2865b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid"; 2875b4aa86bSJames Feist static constexpr const char* pidZoneConfigurationIface = 2885b4aa86bSJames Feist "xyz.openbmc_project.Configuration.Pid.Zone"; 289b7a08d04SJames Feist static constexpr const char* stepwiseConfigurationIface = 290b7a08d04SJames Feist "xyz.openbmc_project.Configuration.Stepwise"; 29173df0db0SJames Feist static constexpr const char* thermalModeIface = 29273df0db0SJames Feist "xyz.openbmc_project.Control.ThermalMode"; 2939c310685SBorawski.Lukasz 2948d1b46d7Szhanghch05 inline void 2958d1b46d7Szhanghch05 asyncPopulatePid(const std::string& connection, const std::string& path, 29673df0db0SJames Feist const std::string& currentProfile, 29773df0db0SJames Feist const std::vector<std::string>& supportedProfiles, 2988d1b46d7Szhanghch05 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 2995b4aa86bSJames Feist { 3005eb468daSGeorge Liu sdbusplus::message::object_path objPath(path); 3015eb468daSGeorge Liu dbus::utility::getManagedObjects( 3025eb468daSGeorge Liu connection, objPath, 30373df0db0SJames Feist [asyncResp, currentProfile, supportedProfiles]( 3045e7e2dc5SEd Tanous const boost::system::error_code& ec, 3055b4aa86bSJames Feist const dbus::utility::ManagedObjectType& managedObj) { 3065b4aa86bSJames Feist if (ec) 3075b4aa86bSJames Feist { 30862598e31SEd Tanous BMCWEB_LOG_ERROR("{}", ec); 309f12894f8SJason M. Bills messages::internalError(asyncResp->res); 3105b4aa86bSJames Feist return; 3115b4aa86bSJames Feist } 3125b4aa86bSJames Feist nlohmann::json& configRoot = 3135b4aa86bSJames Feist asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"]; 3145b4aa86bSJames Feist nlohmann::json& fans = configRoot["FanControllers"]; 3155b4aa86bSJames Feist fans["@odata.type"] = "#OemManager.FanControllers"; 3160fda0f12SGeorge Liu fans["@odata.id"] = 3170fda0f12SGeorge Liu "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanControllers"; 3185b4aa86bSJames Feist 3195b4aa86bSJames Feist nlohmann::json& pids = configRoot["PidControllers"]; 3205b4aa86bSJames Feist pids["@odata.type"] = "#OemManager.PidControllers"; 3215b4aa86bSJames Feist pids["@odata.id"] = 3225b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers"; 3235b4aa86bSJames Feist 324b7a08d04SJames Feist nlohmann::json& stepwise = configRoot["StepwiseControllers"]; 325b7a08d04SJames Feist stepwise["@odata.type"] = "#OemManager.StepwiseControllers"; 326b7a08d04SJames Feist stepwise["@odata.id"] = 327b7a08d04SJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers"; 328b7a08d04SJames Feist 3295b4aa86bSJames Feist nlohmann::json& zones = configRoot["FanZones"]; 3305b4aa86bSJames Feist zones["@odata.id"] = 3315b4aa86bSJames Feist "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones"; 3325b4aa86bSJames Feist zones["@odata.type"] = "#OemManager.FanZones"; 333002d39b4SEd Tanous configRoot["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan"; 3345b4aa86bSJames Feist configRoot["@odata.type"] = "#OemManager.Fan"; 33573df0db0SJames Feist configRoot["Profile@Redfish.AllowableValues"] = supportedProfiles; 33673df0db0SJames Feist 33773df0db0SJames Feist if (!currentProfile.empty()) 33873df0db0SJames Feist { 33973df0db0SJames Feist configRoot["Profile"] = currentProfile; 34073df0db0SJames Feist } 341bf2ddedeSCarson Labrado BMCWEB_LOG_DEBUG("profile = {} !", currentProfile); 3425b4aa86bSJames Feist 3435b4aa86bSJames Feist for (const auto& pathPair : managedObj) 3445b4aa86bSJames Feist { 3455b4aa86bSJames Feist for (const auto& intfPair : pathPair.second) 3465b4aa86bSJames Feist { 3475b4aa86bSJames Feist if (intfPair.first != pidConfigurationIface && 348b7a08d04SJames Feist intfPair.first != pidZoneConfigurationIface && 349b7a08d04SJames Feist intfPair.first != stepwiseConfigurationIface) 3505b4aa86bSJames Feist { 3515b4aa86bSJames Feist continue; 3525b4aa86bSJames Feist } 35373df0db0SJames Feist 354711ac7a9SEd Tanous std::string name; 355711ac7a9SEd Tanous 356711ac7a9SEd Tanous for (const std::pair<std::string, 357002d39b4SEd Tanous dbus::utility::DbusVariantType>& propPair : 358002d39b4SEd Tanous intfPair.second) 359711ac7a9SEd Tanous { 360711ac7a9SEd Tanous if (propPair.first == "Name") 361711ac7a9SEd Tanous { 3625b4aa86bSJames Feist const std::string* namePtr = 363711ac7a9SEd Tanous std::get_if<std::string>(&propPair.second); 3645b4aa86bSJames Feist if (namePtr == nullptr) 3655b4aa86bSJames Feist { 36662598e31SEd Tanous BMCWEB_LOG_ERROR("Pid Name Field illegal"); 367b7a08d04SJames Feist messages::internalError(asyncResp->res); 3685b4aa86bSJames Feist return; 3695b4aa86bSJames Feist } 370db697703SWilly Tu name = *namePtr; 3715b4aa86bSJames Feist dbus::utility::escapePathForDbus(name); 372711ac7a9SEd Tanous } 373711ac7a9SEd Tanous else if (propPair.first == "Profiles") 37473df0db0SJames Feist { 37573df0db0SJames Feist const std::vector<std::string>* profiles = 37673df0db0SJames Feist std::get_if<std::vector<std::string>>( 377711ac7a9SEd Tanous &propPair.second); 37873df0db0SJames Feist if (profiles == nullptr) 37973df0db0SJames Feist { 38062598e31SEd Tanous BMCWEB_LOG_ERROR("Pid Profiles Field illegal"); 38173df0db0SJames Feist messages::internalError(asyncResp->res); 38273df0db0SJames Feist return; 38373df0db0SJames Feist } 38473df0db0SJames Feist if (std::find(profiles->begin(), profiles->end(), 38573df0db0SJames Feist currentProfile) == profiles->end()) 38673df0db0SJames Feist { 38762598e31SEd Tanous BMCWEB_LOG_INFO( 38862598e31SEd Tanous "{} not supported in current profile", name); 38973df0db0SJames Feist continue; 39073df0db0SJames Feist } 39173df0db0SJames Feist } 392711ac7a9SEd Tanous } 393b7a08d04SJames Feist nlohmann::json* config = nullptr; 394c33a90ecSJames Feist const std::string* classPtr = nullptr; 395711ac7a9SEd Tanous 396711ac7a9SEd Tanous for (const std::pair<std::string, 397002d39b4SEd Tanous dbus::utility::DbusVariantType>& propPair : 398002d39b4SEd Tanous intfPair.second) 399c33a90ecSJames Feist { 400727dc83fSLei YU if (propPair.first == "Class") 401711ac7a9SEd Tanous { 402002d39b4SEd Tanous classPtr = std::get_if<std::string>(&propPair.second); 403711ac7a9SEd Tanous } 404c33a90ecSJames Feist } 405c33a90ecSJames Feist 406ef4c65b7SEd Tanous boost::urls::url url("/redfish/v1/Managers/bmc"); 4075b4aa86bSJames Feist if (intfPair.first == pidZoneConfigurationIface) 4085b4aa86bSJames Feist { 4095b4aa86bSJames Feist std::string chassis; 410002d39b4SEd Tanous if (!dbus::utility::getNthStringFromPath(pathPair.first.str, 411002d39b4SEd Tanous 5, chassis)) 4125b4aa86bSJames Feist { 4135b4aa86bSJames Feist chassis = "#IllegalValue"; 4145b4aa86bSJames Feist } 4155b4aa86bSJames Feist nlohmann::json& zone = zones[name]; 416ef4c65b7SEd Tanous zone["Chassis"]["@odata.id"] = 417ef4c65b7SEd Tanous boost::urls::format("/redfish/v1/Chassis/{}", chassis); 418eddfc437SWilly Tu url.set_fragment( 419eddfc437SWilly Tu ("/Oem/OpenBmc/Fan/FanZones"_json_pointer / name) 420eddfc437SWilly Tu .to_string()); 421eddfc437SWilly Tu zone["@odata.id"] = std::move(url); 4225b4aa86bSJames Feist zone["@odata.type"] = "#OemManager.FanZone"; 423b7a08d04SJames Feist config = &zone; 4245b4aa86bSJames Feist } 4255b4aa86bSJames Feist 426b7a08d04SJames Feist else if (intfPair.first == stepwiseConfigurationIface) 4275b4aa86bSJames Feist { 428c33a90ecSJames Feist if (classPtr == nullptr) 429c33a90ecSJames Feist { 43062598e31SEd Tanous BMCWEB_LOG_ERROR("Pid Class Field illegal"); 431c33a90ecSJames Feist messages::internalError(asyncResp->res); 432c33a90ecSJames Feist return; 433c33a90ecSJames Feist } 434c33a90ecSJames Feist 435b7a08d04SJames Feist nlohmann::json& controller = stepwise[name]; 436b7a08d04SJames Feist config = &controller; 437eddfc437SWilly Tu url.set_fragment( 438eddfc437SWilly Tu ("/Oem/OpenBmc/Fan/StepwiseControllers"_json_pointer / 439eddfc437SWilly Tu name) 440eddfc437SWilly Tu .to_string()); 441eddfc437SWilly Tu controller["@odata.id"] = std::move(url); 442b7a08d04SJames Feist controller["@odata.type"] = 443b7a08d04SJames Feist "#OemManager.StepwiseController"; 444b7a08d04SJames Feist 445c33a90ecSJames Feist controller["Direction"] = *classPtr; 4465b4aa86bSJames Feist } 4475b4aa86bSJames Feist 4485b4aa86bSJames Feist // pid and fans are off the same configuration 449b7a08d04SJames Feist else if (intfPair.first == pidConfigurationIface) 4505b4aa86bSJames Feist { 4515b4aa86bSJames Feist if (classPtr == nullptr) 4525b4aa86bSJames Feist { 45362598e31SEd Tanous BMCWEB_LOG_ERROR("Pid Class Field illegal"); 454a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 4555b4aa86bSJames Feist return; 4565b4aa86bSJames Feist } 4575b4aa86bSJames Feist bool isFan = *classPtr == "fan"; 458002d39b4SEd Tanous nlohmann::json& element = isFan ? fans[name] : pids[name]; 459b7a08d04SJames Feist config = &element; 4605b4aa86bSJames Feist if (isFan) 4615b4aa86bSJames Feist { 462eddfc437SWilly Tu url.set_fragment( 463eddfc437SWilly Tu ("/Oem/OpenBmc/Fan/FanControllers"_json_pointer / 464eddfc437SWilly Tu name) 465eddfc437SWilly Tu .to_string()); 466eddfc437SWilly Tu element["@odata.id"] = std::move(url); 467002d39b4SEd Tanous element["@odata.type"] = "#OemManager.FanController"; 4685b4aa86bSJames Feist } 4695b4aa86bSJames Feist else 4705b4aa86bSJames Feist { 471eddfc437SWilly Tu url.set_fragment( 472eddfc437SWilly Tu ("/Oem/OpenBmc/Fan/PidControllers"_json_pointer / 473eddfc437SWilly Tu name) 474eddfc437SWilly Tu .to_string()); 475eddfc437SWilly Tu element["@odata.id"] = std::move(url); 476002d39b4SEd Tanous element["@odata.type"] = "#OemManager.PidController"; 4775b4aa86bSJames Feist } 478b7a08d04SJames Feist } 479b7a08d04SJames Feist else 480b7a08d04SJames Feist { 48162598e31SEd Tanous BMCWEB_LOG_ERROR("Unexpected configuration"); 482b7a08d04SJames Feist messages::internalError(asyncResp->res); 483b7a08d04SJames Feist return; 484b7a08d04SJames Feist } 485b7a08d04SJames Feist 486b7a08d04SJames Feist // used for making maps out of 2 vectors 487b7a08d04SJames Feist const std::vector<double>* keys = nullptr; 488b7a08d04SJames Feist const std::vector<double>* values = nullptr; 489b7a08d04SJames Feist 490b7a08d04SJames Feist for (const auto& propertyPair : intfPair.second) 491b7a08d04SJames Feist { 492b7a08d04SJames Feist if (propertyPair.first == "Type" || 493b7a08d04SJames Feist propertyPair.first == "Class" || 494b7a08d04SJames Feist propertyPair.first == "Name") 495b7a08d04SJames Feist { 496b7a08d04SJames Feist continue; 497b7a08d04SJames Feist } 498b7a08d04SJames Feist 499b7a08d04SJames Feist // zones 500b7a08d04SJames Feist if (intfPair.first == pidZoneConfigurationIface) 501b7a08d04SJames Feist { 502b7a08d04SJames Feist const double* ptr = 503abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 504b7a08d04SJames Feist if (ptr == nullptr) 505b7a08d04SJames Feist { 50662598e31SEd Tanous BMCWEB_LOG_ERROR("Field Illegal {}", 50762598e31SEd Tanous propertyPair.first); 508b7a08d04SJames Feist messages::internalError(asyncResp->res); 509b7a08d04SJames Feist return; 510b7a08d04SJames Feist } 511b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 512b7a08d04SJames Feist } 513b7a08d04SJames Feist 514b7a08d04SJames Feist if (intfPair.first == stepwiseConfigurationIface) 515b7a08d04SJames Feist { 516b7a08d04SJames Feist if (propertyPair.first == "Reading" || 517b7a08d04SJames Feist propertyPair.first == "Output") 518b7a08d04SJames Feist { 519b7a08d04SJames Feist const std::vector<double>* ptr = 520abf2add6SEd Tanous std::get_if<std::vector<double>>( 521b7a08d04SJames Feist &propertyPair.second); 522b7a08d04SJames Feist 523b7a08d04SJames Feist if (ptr == nullptr) 524b7a08d04SJames Feist { 52562598e31SEd Tanous BMCWEB_LOG_ERROR("Field Illegal {}", 52662598e31SEd Tanous propertyPair.first); 527b7a08d04SJames Feist messages::internalError(asyncResp->res); 528b7a08d04SJames Feist return; 529b7a08d04SJames Feist } 530b7a08d04SJames Feist 531b7a08d04SJames Feist if (propertyPair.first == "Reading") 532b7a08d04SJames Feist { 533b7a08d04SJames Feist keys = ptr; 534b7a08d04SJames Feist } 535b7a08d04SJames Feist else 536b7a08d04SJames Feist { 537b7a08d04SJames Feist values = ptr; 538b7a08d04SJames Feist } 539e662eae8SEd Tanous if (keys != nullptr && values != nullptr) 540b7a08d04SJames Feist { 541b7a08d04SJames Feist if (keys->size() != values->size()) 542b7a08d04SJames Feist { 54362598e31SEd Tanous BMCWEB_LOG_ERROR( 54462598e31SEd Tanous "Reading and Output size don't match "); 545b7a08d04SJames Feist messages::internalError(asyncResp->res); 546b7a08d04SJames Feist return; 547b7a08d04SJames Feist } 548b7a08d04SJames Feist nlohmann::json& steps = (*config)["Steps"]; 549b7a08d04SJames Feist steps = nlohmann::json::array(); 550b7a08d04SJames Feist for (size_t ii = 0; ii < keys->size(); ii++) 551b7a08d04SJames Feist { 5521476687dSEd Tanous nlohmann::json::object_t step; 5531476687dSEd Tanous step["Target"] = (*keys)[ii]; 5541476687dSEd Tanous step["Output"] = (*values)[ii]; 555b2ba3072SPatrick Williams steps.emplace_back(std::move(step)); 556b7a08d04SJames Feist } 557b7a08d04SJames Feist } 558b7a08d04SJames Feist } 559b7a08d04SJames Feist if (propertyPair.first == "NegativeHysteresis" || 560b7a08d04SJames Feist propertyPair.first == "PositiveHysteresis") 561b7a08d04SJames Feist { 562b7a08d04SJames Feist const double* ptr = 563abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 564b7a08d04SJames Feist if (ptr == nullptr) 565b7a08d04SJames Feist { 56662598e31SEd Tanous BMCWEB_LOG_ERROR("Field Illegal {}", 56762598e31SEd Tanous propertyPair.first); 568b7a08d04SJames Feist messages::internalError(asyncResp->res); 569b7a08d04SJames Feist return; 570b7a08d04SJames Feist } 571b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 572b7a08d04SJames Feist } 573b7a08d04SJames Feist } 574b7a08d04SJames Feist 575b7a08d04SJames Feist // pid and fans are off the same configuration 576b7a08d04SJames Feist if (intfPair.first == pidConfigurationIface || 577b7a08d04SJames Feist intfPair.first == stepwiseConfigurationIface) 578b7a08d04SJames Feist { 5795b4aa86bSJames Feist if (propertyPair.first == "Zones") 5805b4aa86bSJames Feist { 5815b4aa86bSJames Feist const std::vector<std::string>* inputs = 582abf2add6SEd Tanous std::get_if<std::vector<std::string>>( 5831b6b96c5SEd Tanous &propertyPair.second); 5845b4aa86bSJames Feist 5855b4aa86bSJames Feist if (inputs == nullptr) 5865b4aa86bSJames Feist { 58762598e31SEd Tanous BMCWEB_LOG_ERROR("Zones Pid Field Illegal"); 588a08b46ccSJason M. Bills messages::internalError(asyncResp->res); 5895b4aa86bSJames Feist return; 5905b4aa86bSJames Feist } 591b7a08d04SJames Feist auto& data = (*config)[propertyPair.first]; 5925b4aa86bSJames Feist data = nlohmann::json::array(); 5935b4aa86bSJames Feist for (std::string itemCopy : *inputs) 5945b4aa86bSJames Feist { 5955b4aa86bSJames Feist dbus::utility::escapePathForDbus(itemCopy); 5961476687dSEd Tanous nlohmann::json::object_t input; 597ef4c65b7SEd Tanous boost::urls::url managerUrl = boost::urls::format( 598ef4c65b7SEd Tanous "/redfish/v1/Managers/bmc#{}", 599eddfc437SWilly Tu ("/Oem/OpenBmc/Fan/FanZones"_json_pointer / 600eddfc437SWilly Tu itemCopy) 601eddfc437SWilly Tu .to_string()); 602eddfc437SWilly Tu input["@odata.id"] = std::move(managerUrl); 603b2ba3072SPatrick Williams data.emplace_back(std::move(input)); 6045b4aa86bSJames Feist } 6055b4aa86bSJames Feist } 6065b4aa86bSJames Feist // todo(james): may never happen, but this 6075b4aa86bSJames Feist // assumes configuration data referenced in the 6085b4aa86bSJames Feist // PID config is provided by the same daemon, we 6095b4aa86bSJames Feist // could add another loop to cover all cases, 6105b4aa86bSJames Feist // but I'm okay kicking this can down the road a 6115b4aa86bSJames Feist // bit 6125b4aa86bSJames Feist 6135b4aa86bSJames Feist else if (propertyPair.first == "Inputs" || 6145b4aa86bSJames Feist propertyPair.first == "Outputs") 6155b4aa86bSJames Feist { 616b7a08d04SJames Feist auto& data = (*config)[propertyPair.first]; 6175b4aa86bSJames Feist const std::vector<std::string>* inputs = 618abf2add6SEd Tanous std::get_if<std::vector<std::string>>( 6191b6b96c5SEd Tanous &propertyPair.second); 6205b4aa86bSJames Feist 6215b4aa86bSJames Feist if (inputs == nullptr) 6225b4aa86bSJames Feist { 62362598e31SEd Tanous BMCWEB_LOG_ERROR("Field Illegal {}", 62462598e31SEd Tanous propertyPair.first); 625f12894f8SJason M. Bills messages::internalError(asyncResp->res); 6265b4aa86bSJames Feist return; 6275b4aa86bSJames Feist } 6285b4aa86bSJames Feist data = *inputs; 629b943aaefSJames Feist } 630b943aaefSJames Feist else if (propertyPair.first == "SetPointOffset") 631b943aaefSJames Feist { 632b943aaefSJames Feist const std::string* ptr = 633002d39b4SEd Tanous std::get_if<std::string>(&propertyPair.second); 634b943aaefSJames Feist 635b943aaefSJames Feist if (ptr == nullptr) 636b943aaefSJames Feist { 63762598e31SEd Tanous BMCWEB_LOG_ERROR("Field Illegal {}", 63862598e31SEd Tanous propertyPair.first); 639b943aaefSJames Feist messages::internalError(asyncResp->res); 640b943aaefSJames Feist return; 641b943aaefSJames Feist } 642b943aaefSJames Feist // translate from dbus to redfish 643b943aaefSJames Feist if (*ptr == "WarningHigh") 644b943aaefSJames Feist { 645b943aaefSJames Feist (*config)["SetPointOffset"] = 646b943aaefSJames Feist "UpperThresholdNonCritical"; 647b943aaefSJames Feist } 648b943aaefSJames Feist else if (*ptr == "WarningLow") 649b943aaefSJames Feist { 650b943aaefSJames Feist (*config)["SetPointOffset"] = 651b943aaefSJames Feist "LowerThresholdNonCritical"; 652b943aaefSJames Feist } 653b943aaefSJames Feist else if (*ptr == "CriticalHigh") 654b943aaefSJames Feist { 655b943aaefSJames Feist (*config)["SetPointOffset"] = 656b943aaefSJames Feist "UpperThresholdCritical"; 657b943aaefSJames Feist } 658b943aaefSJames Feist else if (*ptr == "CriticalLow") 659b943aaefSJames Feist { 660b943aaefSJames Feist (*config)["SetPointOffset"] = 661b943aaefSJames Feist "LowerThresholdCritical"; 662b943aaefSJames Feist } 663b943aaefSJames Feist else 664b943aaefSJames Feist { 66562598e31SEd Tanous BMCWEB_LOG_ERROR("Value Illegal {}", *ptr); 666b943aaefSJames Feist messages::internalError(asyncResp->res); 667b943aaefSJames Feist return; 668b943aaefSJames Feist } 669b943aaefSJames Feist } 670b943aaefSJames Feist // doubles 671002d39b4SEd Tanous else if (propertyPair.first == "FFGainCoefficient" || 6725b4aa86bSJames Feist propertyPair.first == "FFOffCoefficient" || 6735b4aa86bSJames Feist propertyPair.first == "ICoefficient" || 6745b4aa86bSJames Feist propertyPair.first == "ILimitMax" || 6755b4aa86bSJames Feist propertyPair.first == "ILimitMin" || 676002d39b4SEd Tanous propertyPair.first == "PositiveHysteresis" || 677002d39b4SEd Tanous propertyPair.first == "NegativeHysteresis" || 6785b4aa86bSJames Feist propertyPair.first == "OutLimitMax" || 6795b4aa86bSJames Feist propertyPair.first == "OutLimitMin" || 6805b4aa86bSJames Feist propertyPair.first == "PCoefficient" || 6817625cb81SJames Feist propertyPair.first == "SetPoint" || 6825b4aa86bSJames Feist propertyPair.first == "SlewNeg" || 6835b4aa86bSJames Feist propertyPair.first == "SlewPos") 6845b4aa86bSJames Feist { 6855b4aa86bSJames Feist const double* ptr = 686abf2add6SEd Tanous std::get_if<double>(&propertyPair.second); 6875b4aa86bSJames Feist if (ptr == nullptr) 6885b4aa86bSJames Feist { 68962598e31SEd Tanous BMCWEB_LOG_ERROR("Field Illegal {}", 69062598e31SEd Tanous propertyPair.first); 691f12894f8SJason M. Bills messages::internalError(asyncResp->res); 6925b4aa86bSJames Feist return; 6935b4aa86bSJames Feist } 694b7a08d04SJames Feist (*config)[propertyPair.first] = *ptr; 6955b4aa86bSJames Feist } 6965b4aa86bSJames Feist } 6975b4aa86bSJames Feist } 6985b4aa86bSJames Feist } 6995b4aa86bSJames Feist } 7005eb468daSGeorge Liu }); 7015b4aa86bSJames Feist } 702ca537928SJennifer Lee 70383ff9ab6SJames Feist enum class CreatePIDRet 70483ff9ab6SJames Feist { 70583ff9ab6SJames Feist fail, 70683ff9ab6SJames Feist del, 70783ff9ab6SJames Feist patch 70883ff9ab6SJames Feist }; 70983ff9ab6SJames Feist 7108d1b46d7Szhanghch05 inline bool 7118d1b46d7Szhanghch05 getZonesFromJsonReq(const std::shared_ptr<bmcweb::AsyncResp>& response, 712*9e9b6049SEd Tanous std::vector<nlohmann::json::object_t>& config, 7135f2caaefSJames Feist std::vector<std::string>& zones) 7145f2caaefSJames Feist { 715b6baeaa4SJames Feist if (config.empty()) 716b6baeaa4SJames Feist { 71762598e31SEd Tanous BMCWEB_LOG_ERROR("Empty Zones"); 718f818b04dSEd Tanous messages::propertyValueFormatError(response->res, config, "Zones"); 719b6baeaa4SJames Feist return false; 720b6baeaa4SJames Feist } 7215f2caaefSJames Feist for (auto& odata : config) 7225f2caaefSJames Feist { 7235f2caaefSJames Feist std::string path; 724*9e9b6049SEd Tanous if (!redfish::json_util::readJsonObject(odata, response->res, 725*9e9b6049SEd Tanous "@odata.id", path)) 7265f2caaefSJames Feist { 7275f2caaefSJames Feist return false; 7285f2caaefSJames Feist } 7295f2caaefSJames Feist std::string input; 73061adbda3SJames Feist 73161adbda3SJames Feist // 8 below comes from 73261adbda3SJames Feist // /redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/Left 73361adbda3SJames Feist // 0 1 2 3 4 5 6 7 8 73461adbda3SJames Feist if (!dbus::utility::getNthStringFromPath(path, 8, input)) 7355f2caaefSJames Feist { 73662598e31SEd Tanous BMCWEB_LOG_ERROR("Got invalid path {}", path); 73762598e31SEd Tanous BMCWEB_LOG_ERROR("Illegal Type Zones"); 738f818b04dSEd Tanous messages::propertyValueFormatError(response->res, odata, "Zones"); 7395f2caaefSJames Feist return false; 7405f2caaefSJames Feist } 741a170f275SEd Tanous std::replace(input.begin(), input.end(), '_', ' '); 7425f2caaefSJames Feist zones.emplace_back(std::move(input)); 7435f2caaefSJames Feist } 7445f2caaefSJames Feist return true; 7455f2caaefSJames Feist } 7465f2caaefSJames Feist 747711ac7a9SEd Tanous inline const dbus::utility::ManagedObjectType::value_type* 74873df0db0SJames Feist findChassis(const dbus::utility::ManagedObjectType& managedObj, 749*9e9b6049SEd Tanous std::string_view value, std::string& chassis) 750b6baeaa4SJames Feist { 75162598e31SEd Tanous BMCWEB_LOG_DEBUG("Find Chassis: {}", value); 752b6baeaa4SJames Feist 753*9e9b6049SEd Tanous std::string escaped(value); 7546ce82fabSYaswanth Reddy M std::replace(escaped.begin(), escaped.end(), ' ', '_'); 755b6baeaa4SJames Feist escaped = "/" + escaped; 7563544d2a7SEd Tanous auto it = std::ranges::find_if(managedObj, [&escaped](const auto& obj) { 75718f8f608SEd Tanous if (obj.first.str.ends_with(escaped)) 758b6baeaa4SJames Feist { 75962598e31SEd Tanous BMCWEB_LOG_DEBUG("Matched {}", obj.first.str); 760b6baeaa4SJames Feist return true; 761b6baeaa4SJames Feist } 762b6baeaa4SJames Feist return false; 763b6baeaa4SJames Feist }); 764b6baeaa4SJames Feist 765b6baeaa4SJames Feist if (it == managedObj.end()) 766b6baeaa4SJames Feist { 76773df0db0SJames Feist return nullptr; 768b6baeaa4SJames Feist } 769b6baeaa4SJames Feist // 5 comes from <chassis-name> being the 5th element 770b6baeaa4SJames Feist // /xyz/openbmc_project/inventory/system/chassis/<chassis-name> 77173df0db0SJames Feist if (dbus::utility::getNthStringFromPath(it->first.str, 5, chassis)) 77273df0db0SJames Feist { 77373df0db0SJames Feist return &(*it); 77473df0db0SJames Feist } 77573df0db0SJames Feist 77673df0db0SJames Feist return nullptr; 777b6baeaa4SJames Feist } 778b6baeaa4SJames Feist 77923a21a1cSEd Tanous inline CreatePIDRet createPidInterface( 7808d1b46d7Szhanghch05 const std::shared_ptr<bmcweb::AsyncResp>& response, const std::string& type, 781*9e9b6049SEd Tanous std::string_view name, nlohmann::json& jsonValue, const std::string& path, 78283ff9ab6SJames Feist const dbus::utility::ManagedObjectType& managedObj, bool createNewObject, 783b9d36b47SEd Tanous dbus::utility::DBusPropertiesMap& output, std::string& chassis, 784b9d36b47SEd Tanous const std::string& profile) 78583ff9ab6SJames Feist { 7865f2caaefSJames Feist // common deleter 787*9e9b6049SEd Tanous if (jsonValue == nullptr) 7885f2caaefSJames Feist { 7895f2caaefSJames Feist std::string iface; 7905f2caaefSJames Feist if (type == "PidControllers" || type == "FanControllers") 7915f2caaefSJames Feist { 7925f2caaefSJames Feist iface = pidConfigurationIface; 7935f2caaefSJames Feist } 7945f2caaefSJames Feist else if (type == "FanZones") 7955f2caaefSJames Feist { 7965f2caaefSJames Feist iface = pidZoneConfigurationIface; 7975f2caaefSJames Feist } 7985f2caaefSJames Feist else if (type == "StepwiseControllers") 7995f2caaefSJames Feist { 8005f2caaefSJames Feist iface = stepwiseConfigurationIface; 8015f2caaefSJames Feist } 8025f2caaefSJames Feist else 8035f2caaefSJames Feist { 80462598e31SEd Tanous BMCWEB_LOG_ERROR("Illegal Type {}", type); 8055f2caaefSJames Feist messages::propertyUnknown(response->res, type); 8065f2caaefSJames Feist return CreatePIDRet::fail; 8075f2caaefSJames Feist } 8086ee7f774SJames Feist 80962598e31SEd Tanous BMCWEB_LOG_DEBUG("del {} {}", path, iface); 8105f2caaefSJames Feist // delete interface 8115f2caaefSJames Feist crow::connections::systemBus->async_method_call( 8125e7e2dc5SEd Tanous [response, path](const boost::system::error_code& ec) { 8135f2caaefSJames Feist if (ec) 8145f2caaefSJames Feist { 81562598e31SEd Tanous BMCWEB_LOG_ERROR("Error patching {}: {}", path, ec); 8165f2caaefSJames Feist messages::internalError(response->res); 817b6baeaa4SJames Feist return; 8185f2caaefSJames Feist } 819b6baeaa4SJames Feist messages::success(response->res); 8205f2caaefSJames Feist }, 8215f2caaefSJames Feist "xyz.openbmc_project.EntityManager", path, iface, "Delete"); 8225f2caaefSJames Feist return CreatePIDRet::del; 8235f2caaefSJames Feist } 8245f2caaefSJames Feist 825711ac7a9SEd Tanous const dbus::utility::ManagedObjectType::value_type* managedItem = nullptr; 826b6baeaa4SJames Feist if (!createNewObject) 827b6baeaa4SJames Feist { 828b6baeaa4SJames Feist // if we aren't creating a new object, we should be able to find it on 829b6baeaa4SJames Feist // d-bus 830*9e9b6049SEd Tanous managedItem = findChassis(managedObj, name, chassis); 83173df0db0SJames Feist if (managedItem == nullptr) 832b6baeaa4SJames Feist { 83362598e31SEd Tanous BMCWEB_LOG_ERROR("Failed to get chassis from config patch"); 834ef4c65b7SEd Tanous messages::invalidObject( 835ef4c65b7SEd Tanous response->res, 836ef4c65b7SEd Tanous boost::urls::format("/redfish/v1/Chassis/{}", chassis)); 837b6baeaa4SJames Feist return CreatePIDRet::fail; 838b6baeaa4SJames Feist } 839b6baeaa4SJames Feist } 840b6baeaa4SJames Feist 84126f6976fSEd Tanous if (!profile.empty() && 84273df0db0SJames Feist (type == "PidControllers" || type == "FanControllers" || 84373df0db0SJames Feist type == "StepwiseControllers")) 84473df0db0SJames Feist { 84573df0db0SJames Feist if (managedItem == nullptr) 84673df0db0SJames Feist { 847b9d36b47SEd Tanous output.emplace_back("Profiles", std::vector<std::string>{profile}); 84873df0db0SJames Feist } 84973df0db0SJames Feist else 85073df0db0SJames Feist { 85173df0db0SJames Feist std::string interface; 85273df0db0SJames Feist if (type == "StepwiseControllers") 85373df0db0SJames Feist { 85473df0db0SJames Feist interface = stepwiseConfigurationIface; 85573df0db0SJames Feist } 85673df0db0SJames Feist else 85773df0db0SJames Feist { 85873df0db0SJames Feist interface = pidConfigurationIface; 85973df0db0SJames Feist } 860711ac7a9SEd Tanous bool ifaceFound = false; 861711ac7a9SEd Tanous for (const auto& iface : managedItem->second) 862711ac7a9SEd Tanous { 863711ac7a9SEd Tanous if (iface.first == interface) 864711ac7a9SEd Tanous { 865711ac7a9SEd Tanous ifaceFound = true; 866711ac7a9SEd Tanous for (const auto& prop : iface.second) 867711ac7a9SEd Tanous { 868711ac7a9SEd Tanous if (prop.first == "Profiles") 869711ac7a9SEd Tanous { 870711ac7a9SEd Tanous const std::vector<std::string>* curProfiles = 871711ac7a9SEd Tanous std::get_if<std::vector<std::string>>( 872711ac7a9SEd Tanous &(prop.second)); 873711ac7a9SEd Tanous if (curProfiles == nullptr) 874711ac7a9SEd Tanous { 87562598e31SEd Tanous BMCWEB_LOG_ERROR( 87662598e31SEd Tanous "Illegal profiles in managed object"); 877711ac7a9SEd Tanous messages::internalError(response->res); 878711ac7a9SEd Tanous return CreatePIDRet::fail; 879711ac7a9SEd Tanous } 880711ac7a9SEd Tanous if (std::find(curProfiles->begin(), 881711ac7a9SEd Tanous curProfiles->end(), 882711ac7a9SEd Tanous profile) == curProfiles->end()) 883711ac7a9SEd Tanous { 884711ac7a9SEd Tanous std::vector<std::string> newProfiles = 885711ac7a9SEd Tanous *curProfiles; 886711ac7a9SEd Tanous newProfiles.push_back(profile); 887b9d36b47SEd Tanous output.emplace_back("Profiles", newProfiles); 888711ac7a9SEd Tanous } 889711ac7a9SEd Tanous } 890711ac7a9SEd Tanous } 891711ac7a9SEd Tanous } 892711ac7a9SEd Tanous } 893711ac7a9SEd Tanous 894711ac7a9SEd Tanous if (!ifaceFound) 89573df0db0SJames Feist { 89662598e31SEd Tanous BMCWEB_LOG_ERROR("Failed to find interface in managed object"); 89773df0db0SJames Feist messages::internalError(response->res); 89873df0db0SJames Feist return CreatePIDRet::fail; 89973df0db0SJames Feist } 90073df0db0SJames Feist } 90173df0db0SJames Feist } 90273df0db0SJames Feist 90383ff9ab6SJames Feist if (type == "PidControllers" || type == "FanControllers") 90483ff9ab6SJames Feist { 90583ff9ab6SJames Feist if (createNewObject) 90683ff9ab6SJames Feist { 907b9d36b47SEd Tanous output.emplace_back("Class", 908b9d36b47SEd Tanous type == "PidControllers" ? "temp" : "fan"); 909b9d36b47SEd Tanous output.emplace_back("Type", "Pid"); 91083ff9ab6SJames Feist } 9115f2caaefSJames Feist 912*9e9b6049SEd Tanous std::optional<std::vector<nlohmann::json::object_t>> zones; 9135f2caaefSJames Feist std::optional<std::vector<std::string>> inputs; 9145f2caaefSJames Feist std::optional<std::vector<std::string>> outputs; 9155f2caaefSJames Feist std::map<std::string, std::optional<double>> doubles; 916b943aaefSJames Feist std::optional<std::string> setpointOffset; 9175f2caaefSJames Feist if (!redfish::json_util::readJson( 918*9e9b6049SEd Tanous jsonValue, response->res, "Inputs", inputs, "Outputs", outputs, 9195f2caaefSJames Feist "Zones", zones, "FFGainCoefficient", 9205f2caaefSJames Feist doubles["FFGainCoefficient"], "FFOffCoefficient", 9215f2caaefSJames Feist doubles["FFOffCoefficient"], "ICoefficient", 9225f2caaefSJames Feist doubles["ICoefficient"], "ILimitMax", doubles["ILimitMax"], 9235f2caaefSJames Feist "ILimitMin", doubles["ILimitMin"], "OutLimitMax", 9245f2caaefSJames Feist doubles["OutLimitMax"], "OutLimitMin", doubles["OutLimitMin"], 9255f2caaefSJames Feist "PCoefficient", doubles["PCoefficient"], "SetPoint", 926b943aaefSJames Feist doubles["SetPoint"], "SetPointOffset", setpointOffset, 927b943aaefSJames Feist "SlewNeg", doubles["SlewNeg"], "SlewPos", doubles["SlewPos"], 928b943aaefSJames Feist "PositiveHysteresis", doubles["PositiveHysteresis"], 929b943aaefSJames Feist "NegativeHysteresis", doubles["NegativeHysteresis"])) 93083ff9ab6SJames Feist { 9315f2caaefSJames Feist return CreatePIDRet::fail; 93283ff9ab6SJames Feist } 9335f2caaefSJames Feist if (zones) 9345f2caaefSJames Feist { 9355f2caaefSJames Feist std::vector<std::string> zonesStr; 9365f2caaefSJames Feist if (!getZonesFromJsonReq(response, *zones, zonesStr)) 9375f2caaefSJames Feist { 93862598e31SEd Tanous BMCWEB_LOG_ERROR("Illegal Zones"); 9395f2caaefSJames Feist return CreatePIDRet::fail; 9405f2caaefSJames Feist } 941b6baeaa4SJames Feist if (chassis.empty() && 942e662eae8SEd Tanous findChassis(managedObj, zonesStr[0], chassis) == nullptr) 943b6baeaa4SJames Feist { 94462598e31SEd Tanous BMCWEB_LOG_ERROR("Failed to get chassis from config patch"); 945ace85d60SEd Tanous messages::invalidObject( 946ef4c65b7SEd Tanous response->res, 947ef4c65b7SEd Tanous boost::urls::format("/redfish/v1/Chassis/{}", chassis)); 948b6baeaa4SJames Feist return CreatePIDRet::fail; 949b6baeaa4SJames Feist } 950b9d36b47SEd Tanous output.emplace_back("Zones", std::move(zonesStr)); 9515f2caaefSJames Feist } 952afb9ee06SEd Tanous 953afb9ee06SEd Tanous if (inputs) 9545f2caaefSJames Feist { 955afb9ee06SEd Tanous for (std::string& value : *inputs) 95683ff9ab6SJames Feist { 957a170f275SEd Tanous std::replace(value.begin(), value.end(), '_', ' '); 95883ff9ab6SJames Feist } 959afb9ee06SEd Tanous output.emplace_back("Inputs", *inputs); 960afb9ee06SEd Tanous } 961afb9ee06SEd Tanous 962afb9ee06SEd Tanous if (outputs) 9635f2caaefSJames Feist { 964afb9ee06SEd Tanous for (std::string& value : *outputs) 9655f2caaefSJames Feist { 966afb9ee06SEd Tanous std::replace(value.begin(), value.end(), '_', ' '); 9675f2caaefSJames Feist } 968afb9ee06SEd Tanous output.emplace_back("Outputs", *outputs); 96983ff9ab6SJames Feist } 97083ff9ab6SJames Feist 971b943aaefSJames Feist if (setpointOffset) 972b943aaefSJames Feist { 973b943aaefSJames Feist // translate between redfish and dbus names 974b943aaefSJames Feist if (*setpointOffset == "UpperThresholdNonCritical") 975b943aaefSJames Feist { 976b9d36b47SEd Tanous output.emplace_back("SetPointOffset", "WarningLow"); 977b943aaefSJames Feist } 978b943aaefSJames Feist else if (*setpointOffset == "LowerThresholdNonCritical") 979b943aaefSJames Feist { 980b9d36b47SEd Tanous output.emplace_back("SetPointOffset", "WarningHigh"); 981b943aaefSJames Feist } 982b943aaefSJames Feist else if (*setpointOffset == "LowerThresholdCritical") 983b943aaefSJames Feist { 984b9d36b47SEd Tanous output.emplace_back("SetPointOffset", "CriticalLow"); 985b943aaefSJames Feist } 986b943aaefSJames Feist else if (*setpointOffset == "UpperThresholdCritical") 987b943aaefSJames Feist { 988b9d36b47SEd Tanous output.emplace_back("SetPointOffset", "CriticalHigh"); 989b943aaefSJames Feist } 990b943aaefSJames Feist else 991b943aaefSJames Feist { 99262598e31SEd Tanous BMCWEB_LOG_ERROR("Invalid setpointoffset {}", *setpointOffset); 993*9e9b6049SEd Tanous messages::propertyValueNotInList(response->res, name, 994ace85d60SEd Tanous "SetPointOffset"); 995b943aaefSJames Feist return CreatePIDRet::fail; 996b943aaefSJames Feist } 997b943aaefSJames Feist } 998b943aaefSJames Feist 99983ff9ab6SJames Feist // doubles 10005f2caaefSJames Feist for (const auto& pairs : doubles) 100183ff9ab6SJames Feist { 10025f2caaefSJames Feist if (!pairs.second) 100383ff9ab6SJames Feist { 10045f2caaefSJames Feist continue; 100583ff9ab6SJames Feist } 100662598e31SEd Tanous BMCWEB_LOG_DEBUG("{} = {}", pairs.first, *pairs.second); 1007b9d36b47SEd Tanous output.emplace_back(pairs.first, *pairs.second); 10085f2caaefSJames Feist } 100983ff9ab6SJames Feist } 101083ff9ab6SJames Feist 101183ff9ab6SJames Feist else if (type == "FanZones") 101283ff9ab6SJames Feist { 1013b9d36b47SEd Tanous output.emplace_back("Type", "Pid.Zone"); 101483ff9ab6SJames Feist 1015*9e9b6049SEd Tanous std::optional<std::string> chassisId; 10165f2caaefSJames Feist std::optional<double> failSafePercent; 1017d3ec07f8SJames Feist std::optional<double> minThermalOutput; 1018*9e9b6049SEd Tanous if (!redfish::json_util::readJson(jsonValue, response->res, 1019*9e9b6049SEd Tanous "Chassis/@odata.id", chassisId, 1020*9e9b6049SEd Tanous "FailSafePercent", failSafePercent, 1021*9e9b6049SEd Tanous "MinThermalOutput", minThermalOutput)) 102283ff9ab6SJames Feist { 102383ff9ab6SJames Feist return CreatePIDRet::fail; 102483ff9ab6SJames Feist } 10255f2caaefSJames Feist 1026*9e9b6049SEd Tanous if (chassisId) 102783ff9ab6SJames Feist { 1028717794d5SAppaRao Puli // /redfish/v1/chassis/chassis_name/ 1029*9e9b6049SEd Tanous if (!dbus::utility::getNthStringFromPath(*chassisId, 3, chassis)) 103083ff9ab6SJames Feist { 1031*9e9b6049SEd Tanous BMCWEB_LOG_ERROR("Got invalid path {}", *chassisId); 1032ace85d60SEd Tanous messages::invalidObject( 1033ef4c65b7SEd Tanous response->res, 1034*9e9b6049SEd Tanous boost::urls::format("/redfish/v1/Chassis/{}", *chassisId)); 103583ff9ab6SJames Feist return CreatePIDRet::fail; 103683ff9ab6SJames Feist } 103783ff9ab6SJames Feist } 1038d3ec07f8SJames Feist if (minThermalOutput) 103983ff9ab6SJames Feist { 1040b9d36b47SEd Tanous output.emplace_back("MinThermalOutput", *minThermalOutput); 10415f2caaefSJames Feist } 10425f2caaefSJames Feist if (failSafePercent) 104383ff9ab6SJames Feist { 1044b9d36b47SEd Tanous output.emplace_back("FailSafePercent", *failSafePercent); 10455f2caaefSJames Feist } 10465f2caaefSJames Feist } 10475f2caaefSJames Feist else if (type == "StepwiseControllers") 10485f2caaefSJames Feist { 1049b9d36b47SEd Tanous output.emplace_back("Type", "Stepwise"); 10505f2caaefSJames Feist 1051*9e9b6049SEd Tanous std::optional<std::vector<nlohmann::json::object_t>> zones; 1052*9e9b6049SEd Tanous std::optional<std::vector<nlohmann::json::object_t>> steps; 10535f2caaefSJames Feist std::optional<std::vector<std::string>> inputs; 10545f2caaefSJames Feist std::optional<double> positiveHysteresis; 10555f2caaefSJames Feist std::optional<double> negativeHysteresis; 1056c33a90ecSJames Feist std::optional<std::string> direction; // upper clipping curve vs lower 10575f2caaefSJames Feist if (!redfish::json_util::readJson( 1058*9e9b6049SEd Tanous jsonValue, response->res, "Zones", zones, "Steps", steps, 1059b6baeaa4SJames Feist "Inputs", inputs, "PositiveHysteresis", positiveHysteresis, 1060c33a90ecSJames Feist "NegativeHysteresis", negativeHysteresis, "Direction", 1061c33a90ecSJames Feist direction)) 10625f2caaefSJames Feist { 106383ff9ab6SJames Feist return CreatePIDRet::fail; 106483ff9ab6SJames Feist } 10655f2caaefSJames Feist 10665f2caaefSJames Feist if (zones) 106783ff9ab6SJames Feist { 1068b6baeaa4SJames Feist std::vector<std::string> zonesStrs; 1069b6baeaa4SJames Feist if (!getZonesFromJsonReq(response, *zones, zonesStrs)) 10705f2caaefSJames Feist { 107162598e31SEd Tanous BMCWEB_LOG_ERROR("Illegal Zones"); 107283ff9ab6SJames Feist return CreatePIDRet::fail; 107383ff9ab6SJames Feist } 1074b6baeaa4SJames Feist if (chassis.empty() && 1075e662eae8SEd Tanous findChassis(managedObj, zonesStrs[0], chassis) == nullptr) 1076b6baeaa4SJames Feist { 107762598e31SEd Tanous BMCWEB_LOG_ERROR("Failed to get chassis from config patch"); 1078ace85d60SEd Tanous messages::invalidObject( 1079ef4c65b7SEd Tanous response->res, 1080ef4c65b7SEd Tanous boost::urls::format("/redfish/v1/Chassis/{}", chassis)); 1081b6baeaa4SJames Feist return CreatePIDRet::fail; 1082b6baeaa4SJames Feist } 1083b9d36b47SEd Tanous output.emplace_back("Zones", std::move(zonesStrs)); 10845f2caaefSJames Feist } 10855f2caaefSJames Feist if (steps) 10865f2caaefSJames Feist { 10875f2caaefSJames Feist std::vector<double> readings; 10885f2caaefSJames Feist std::vector<double> outputs; 10895f2caaefSJames Feist for (auto& step : *steps) 10905f2caaefSJames Feist { 1091543f4400SEd Tanous double target = 0.0; 1092543f4400SEd Tanous double out = 0.0; 10935f2caaefSJames Feist 1094*9e9b6049SEd Tanous if (!redfish::json_util::readJsonObject( 1095*9e9b6049SEd Tanous step, response->res, "Target", target, "Output", out)) 10965f2caaefSJames Feist { 10975f2caaefSJames Feist return CreatePIDRet::fail; 10985f2caaefSJames Feist } 10995f2caaefSJames Feist readings.emplace_back(target); 110023a21a1cSEd Tanous outputs.emplace_back(out); 11015f2caaefSJames Feist } 1102b9d36b47SEd Tanous output.emplace_back("Reading", std::move(readings)); 1103b9d36b47SEd Tanous output.emplace_back("Output", std::move(outputs)); 11045f2caaefSJames Feist } 11055f2caaefSJames Feist if (inputs) 11065f2caaefSJames Feist { 11075f2caaefSJames Feist for (std::string& value : *inputs) 11085f2caaefSJames Feist { 1109a170f275SEd Tanous std::replace(value.begin(), value.end(), '_', ' '); 11105f2caaefSJames Feist } 1111b9d36b47SEd Tanous output.emplace_back("Inputs", std::move(*inputs)); 11125f2caaefSJames Feist } 11135f2caaefSJames Feist if (negativeHysteresis) 11145f2caaefSJames Feist { 1115b9d36b47SEd Tanous output.emplace_back("NegativeHysteresis", *negativeHysteresis); 11165f2caaefSJames Feist } 11175f2caaefSJames Feist if (positiveHysteresis) 11185f2caaefSJames Feist { 1119b9d36b47SEd Tanous output.emplace_back("PositiveHysteresis", *positiveHysteresis); 112083ff9ab6SJames Feist } 1121c33a90ecSJames Feist if (direction) 1122c33a90ecSJames Feist { 1123c33a90ecSJames Feist constexpr const std::array<const char*, 2> allowedDirections = { 1124c33a90ecSJames Feist "Ceiling", "Floor"}; 11253544d2a7SEd Tanous if (std::ranges::find(allowedDirections, *direction) == 11263544d2a7SEd Tanous allowedDirections.end()) 1127c33a90ecSJames Feist { 1128c33a90ecSJames Feist messages::propertyValueTypeError(response->res, "Direction", 1129c33a90ecSJames Feist *direction); 1130c33a90ecSJames Feist return CreatePIDRet::fail; 1131c33a90ecSJames Feist } 1132b9d36b47SEd Tanous output.emplace_back("Class", *direction); 1133c33a90ecSJames Feist } 113483ff9ab6SJames Feist } 113583ff9ab6SJames Feist else 113683ff9ab6SJames Feist { 113762598e31SEd Tanous BMCWEB_LOG_ERROR("Illegal Type {}", type); 113835a62c7cSJason M. Bills messages::propertyUnknown(response->res, type); 113983ff9ab6SJames Feist return CreatePIDRet::fail; 114083ff9ab6SJames Feist } 114183ff9ab6SJames Feist return CreatePIDRet::patch; 114283ff9ab6SJames Feist } 114373df0db0SJames Feist struct GetPIDValues : std::enable_shared_from_this<GetPIDValues> 114473df0db0SJames Feist { 11456936afe4SEd Tanous struct CompletionValues 11466936afe4SEd Tanous { 11476936afe4SEd Tanous std::vector<std::string> supportedProfiles; 11486936afe4SEd Tanous std::string currentProfile; 11496936afe4SEd Tanous dbus::utility::MapperGetSubTreeResponse subtree; 11506936afe4SEd Tanous }; 115183ff9ab6SJames Feist 11524e23a444SEd Tanous explicit GetPIDValues( 11534e23a444SEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn) : 115423a21a1cSEd Tanous asyncResp(asyncRespIn) 115573df0db0SJames Feist 11561214b7e7SGunnar Mills {} 11579c310685SBorawski.Lukasz 115873df0db0SJames Feist void run() 11595b4aa86bSJames Feist { 116073df0db0SJames Feist std::shared_ptr<GetPIDValues> self = shared_from_this(); 116173df0db0SJames Feist 116273df0db0SJames Feist // get all configurations 1163e99073f5SGeorge Liu constexpr std::array<std::string_view, 4> interfaces = { 1164e99073f5SGeorge Liu pidConfigurationIface, pidZoneConfigurationIface, 1165e99073f5SGeorge Liu objectManagerIface, stepwiseConfigurationIface}; 1166e99073f5SGeorge Liu dbus::utility::getSubTree( 1167e99073f5SGeorge Liu "/", 0, interfaces, 1168b9d36b47SEd Tanous [self]( 1169e99073f5SGeorge Liu const boost::system::error_code& ec, 1170b9d36b47SEd Tanous const dbus::utility::MapperGetSubTreeResponse& subtreeLocal) { 11715b4aa86bSJames Feist if (ec) 11725b4aa86bSJames Feist { 117362598e31SEd Tanous BMCWEB_LOG_ERROR("{}", ec); 117473df0db0SJames Feist messages::internalError(self->asyncResp->res); 117573df0db0SJames Feist return; 117673df0db0SJames Feist } 11776936afe4SEd Tanous self->complete.subtree = subtreeLocal; 1178e99073f5SGeorge Liu }); 117973df0db0SJames Feist 118073df0db0SJames Feist // at the same time get the selected profile 1181e99073f5SGeorge Liu constexpr std::array<std::string_view, 1> thermalModeIfaces = { 1182e99073f5SGeorge Liu thermalModeIface}; 1183e99073f5SGeorge Liu dbus::utility::getSubTree( 1184e99073f5SGeorge Liu "/", 0, thermalModeIfaces, 1185b9d36b47SEd Tanous [self]( 1186e99073f5SGeorge Liu const boost::system::error_code& ec, 1187b9d36b47SEd Tanous const dbus::utility::MapperGetSubTreeResponse& subtreeLocal) { 118823a21a1cSEd Tanous if (ec || subtreeLocal.empty()) 118973df0db0SJames Feist { 119073df0db0SJames Feist return; 119173df0db0SJames Feist } 119223a21a1cSEd Tanous if (subtreeLocal[0].second.size() != 1) 119373df0db0SJames Feist { 119473df0db0SJames Feist // invalid mapper response, should never happen 119562598e31SEd Tanous BMCWEB_LOG_ERROR("GetPIDValues: Mapper Error"); 119673df0db0SJames Feist messages::internalError(self->asyncResp->res); 11975b4aa86bSJames Feist return; 11985b4aa86bSJames Feist } 11995b4aa86bSJames Feist 120023a21a1cSEd Tanous const std::string& path = subtreeLocal[0].first; 120123a21a1cSEd Tanous const std::string& owner = subtreeLocal[0].second[0].first; 1202fac6e53bSKrzysztof Grobelny 1203fac6e53bSKrzysztof Grobelny sdbusplus::asio::getAllProperties( 1204fac6e53bSKrzysztof Grobelny *crow::connections::systemBus, owner, path, thermalModeIface, 1205168e20c1SEd Tanous [path, owner, 12065e7e2dc5SEd Tanous self](const boost::system::error_code& ec2, 1207b9d36b47SEd Tanous const dbus::utility::DBusPropertiesMap& resp) { 120823a21a1cSEd Tanous if (ec2) 120973df0db0SJames Feist { 121062598e31SEd Tanous BMCWEB_LOG_ERROR( 121162598e31SEd Tanous "GetPIDValues: Can't get thermalModeIface {}", path); 121273df0db0SJames Feist messages::internalError(self->asyncResp->res); 121373df0db0SJames Feist return; 121473df0db0SJames Feist } 1215fac6e53bSKrzysztof Grobelny 1216271584abSEd Tanous const std::string* current = nullptr; 1217271584abSEd Tanous const std::vector<std::string>* supported = nullptr; 1218fac6e53bSKrzysztof Grobelny 1219fac6e53bSKrzysztof Grobelny const bool success = sdbusplus::unpackPropertiesNoThrow( 1220fac6e53bSKrzysztof Grobelny dbus_utils::UnpackErrorPrinter(), resp, "Current", current, 1221fac6e53bSKrzysztof Grobelny "Supported", supported); 1222fac6e53bSKrzysztof Grobelny 1223fac6e53bSKrzysztof Grobelny if (!success) 122473df0db0SJames Feist { 1225002d39b4SEd Tanous messages::internalError(self->asyncResp->res); 122673df0db0SJames Feist return; 122773df0db0SJames Feist } 1228fac6e53bSKrzysztof Grobelny 122973df0db0SJames Feist if (current == nullptr || supported == nullptr) 123073df0db0SJames Feist { 123162598e31SEd Tanous BMCWEB_LOG_ERROR( 123262598e31SEd Tanous "GetPIDValues: thermal mode iface invalid {}", path); 123373df0db0SJames Feist messages::internalError(self->asyncResp->res); 123473df0db0SJames Feist return; 123573df0db0SJames Feist } 12366936afe4SEd Tanous self->complete.currentProfile = *current; 12376936afe4SEd Tanous self->complete.supportedProfiles = *supported; 1238fac6e53bSKrzysztof Grobelny }); 1239e99073f5SGeorge Liu }); 124073df0db0SJames Feist } 124173df0db0SJames Feist 12426936afe4SEd Tanous static void 12436936afe4SEd Tanous processingComplete(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 12446936afe4SEd Tanous const CompletionValues& completion) 124573df0db0SJames Feist { 124673df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 124773df0db0SJames Feist { 124873df0db0SJames Feist return; 124973df0db0SJames Feist } 12505b4aa86bSJames Feist // create map of <connection, path to objMgr>> 12516936afe4SEd Tanous boost::container::flat_map< 12526936afe4SEd Tanous std::string, std::string, std::less<>, 12536936afe4SEd Tanous std::vector<std::pair<std::string, std::string>>> 12546936afe4SEd Tanous objectMgrPaths; 12556936afe4SEd Tanous boost::container::flat_set<std::string, std::less<>, 12566936afe4SEd Tanous std::vector<std::string>> 12576936afe4SEd Tanous calledConnections; 12586936afe4SEd Tanous for (const auto& pathGroup : completion.subtree) 12595b4aa86bSJames Feist { 12605b4aa86bSJames Feist for (const auto& connectionGroup : pathGroup.second) 12615b4aa86bSJames Feist { 12626bce33bcSJames Feist auto findConnection = 12636bce33bcSJames Feist calledConnections.find(connectionGroup.first); 12646bce33bcSJames Feist if (findConnection != calledConnections.end()) 12656bce33bcSJames Feist { 12666bce33bcSJames Feist break; 12676bce33bcSJames Feist } 126873df0db0SJames Feist for (const std::string& interface : connectionGroup.second) 12695b4aa86bSJames Feist { 12705b4aa86bSJames Feist if (interface == objectManagerIface) 12715b4aa86bSJames Feist { 127273df0db0SJames Feist objectMgrPaths[connectionGroup.first] = pathGroup.first; 12735b4aa86bSJames Feist } 12745b4aa86bSJames Feist // this list is alphabetical, so we 12755b4aa86bSJames Feist // should have found the objMgr by now 12765b4aa86bSJames Feist if (interface == pidConfigurationIface || 1277b7a08d04SJames Feist interface == pidZoneConfigurationIface || 1278b7a08d04SJames Feist interface == stepwiseConfigurationIface) 12795b4aa86bSJames Feist { 12805b4aa86bSJames Feist auto findObjMgr = 12815b4aa86bSJames Feist objectMgrPaths.find(connectionGroup.first); 12825b4aa86bSJames Feist if (findObjMgr == objectMgrPaths.end()) 12835b4aa86bSJames Feist { 128462598e31SEd Tanous BMCWEB_LOG_DEBUG("{}Has no Object Manager", 128562598e31SEd Tanous connectionGroup.first); 12865b4aa86bSJames Feist continue; 12875b4aa86bSJames Feist } 12886bce33bcSJames Feist 12896bce33bcSJames Feist calledConnections.insert(connectionGroup.first); 12906bce33bcSJames Feist 129173df0db0SJames Feist asyncPopulatePid(findObjMgr->first, findObjMgr->second, 12926936afe4SEd Tanous completion.currentProfile, 12936936afe4SEd Tanous completion.supportedProfiles, 129473df0db0SJames Feist asyncResp); 12955b4aa86bSJames Feist break; 12965b4aa86bSJames Feist } 12975b4aa86bSJames Feist } 12985b4aa86bSJames Feist } 12995b4aa86bSJames Feist } 130073df0db0SJames Feist } 130173df0db0SJames Feist 13026936afe4SEd Tanous ~GetPIDValues() 13036936afe4SEd Tanous { 13046936afe4SEd Tanous boost::asio::post(crow::connections::systemBus->get_io_context(), 13056936afe4SEd Tanous std::bind_front(&processingComplete, asyncResp, 13066936afe4SEd Tanous std::move(complete))); 13076936afe4SEd Tanous } 13086936afe4SEd Tanous 1309ecd6a3a2SEd Tanous GetPIDValues(const GetPIDValues&) = delete; 1310ecd6a3a2SEd Tanous GetPIDValues(GetPIDValues&&) = delete; 1311ecd6a3a2SEd Tanous GetPIDValues& operator=(const GetPIDValues&) = delete; 1312ecd6a3a2SEd Tanous GetPIDValues& operator=(GetPIDValues&&) = delete; 1313ecd6a3a2SEd Tanous 13148d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> asyncResp; 13156936afe4SEd Tanous CompletionValues complete; 131673df0db0SJames Feist }; 131773df0db0SJames Feist 131873df0db0SJames Feist struct SetPIDValues : std::enable_shared_from_this<SetPIDValues> 131973df0db0SJames Feist { 1320*9e9b6049SEd Tanous SetPIDValues( 1321*9e9b6049SEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn, 1322*9e9b6049SEd Tanous std::vector< 1323*9e9b6049SEd Tanous std::pair<std::string, std::optional<nlohmann::json::object_t>>>&& 1324*9e9b6049SEd Tanous configurationsIn, 1325*9e9b6049SEd Tanous std::optional<std::string>& profileIn) : 1326*9e9b6049SEd Tanous asyncResp(asyncRespIn), 1327*9e9b6049SEd Tanous configuration(std::move(configurationsIn)), 1328*9e9b6049SEd Tanous profile(std::move(profileIn)) 1329*9e9b6049SEd Tanous {} 1330ecd6a3a2SEd Tanous 1331ecd6a3a2SEd Tanous SetPIDValues(const SetPIDValues&) = delete; 1332ecd6a3a2SEd Tanous SetPIDValues(SetPIDValues&&) = delete; 1333ecd6a3a2SEd Tanous SetPIDValues& operator=(const SetPIDValues&) = delete; 1334ecd6a3a2SEd Tanous SetPIDValues& operator=(SetPIDValues&&) = delete; 1335ecd6a3a2SEd Tanous 133673df0db0SJames Feist void run() 133773df0db0SJames Feist { 133873df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 133973df0db0SJames Feist { 134073df0db0SJames Feist return; 134173df0db0SJames Feist } 134273df0db0SJames Feist 134373df0db0SJames Feist std::shared_ptr<SetPIDValues> self = shared_from_this(); 134473df0db0SJames Feist 134573df0db0SJames Feist // todo(james): might make sense to do a mapper call here if this 134673df0db0SJames Feist // interface gets more traction 13475eb468daSGeorge Liu sdbusplus::message::object_path objPath( 13485eb468daSGeorge Liu "/xyz/openbmc_project/inventory"); 13495eb468daSGeorge Liu dbus::utility::getManagedObjects( 13505eb468daSGeorge Liu "xyz.openbmc_project.EntityManager", objPath, 13515e7e2dc5SEd Tanous [self](const boost::system::error_code& ec, 1352914e2d5dSEd Tanous const dbus::utility::ManagedObjectType& mObj) { 135373df0db0SJames Feist if (ec) 135473df0db0SJames Feist { 135562598e31SEd Tanous BMCWEB_LOG_ERROR("Error communicating to Entity Manager"); 135673df0db0SJames Feist messages::internalError(self->asyncResp->res); 135773df0db0SJames Feist return; 135873df0db0SJames Feist } 1359e69d9de2SJames Feist const std::array<const char*, 3> configurations = { 1360e69d9de2SJames Feist pidConfigurationIface, pidZoneConfigurationIface, 1361e69d9de2SJames Feist stepwiseConfigurationIface}; 1362e69d9de2SJames Feist 136314b0b8d5SJames Feist for (const auto& [path, object] : mObj) 1364e69d9de2SJames Feist { 136514b0b8d5SJames Feist for (const auto& [interface, _] : object) 1366e69d9de2SJames Feist { 13673544d2a7SEd Tanous if (std::ranges::find(configurations, interface) != 13683544d2a7SEd Tanous configurations.end()) 1369e69d9de2SJames Feist { 137014b0b8d5SJames Feist self->objectCount++; 1371e69d9de2SJames Feist break; 1372e69d9de2SJames Feist } 1373e69d9de2SJames Feist } 1374e69d9de2SJames Feist } 1375914e2d5dSEd Tanous self->managedObj = mObj; 13765eb468daSGeorge Liu }); 137773df0db0SJames Feist 137873df0db0SJames Feist // at the same time get the profile information 1379e99073f5SGeorge Liu constexpr std::array<std::string_view, 1> thermalModeIfaces = { 1380e99073f5SGeorge Liu thermalModeIface}; 1381e99073f5SGeorge Liu dbus::utility::getSubTree( 1382e99073f5SGeorge Liu "/", 0, thermalModeIfaces, 1383e99073f5SGeorge Liu [self](const boost::system::error_code& ec, 1384b9d36b47SEd Tanous const dbus::utility::MapperGetSubTreeResponse& subtree) { 138573df0db0SJames Feist if (ec || subtree.empty()) 138673df0db0SJames Feist { 138773df0db0SJames Feist return; 138873df0db0SJames Feist } 138973df0db0SJames Feist if (subtree[0].second.empty()) 139073df0db0SJames Feist { 139173df0db0SJames Feist // invalid mapper response, should never happen 139262598e31SEd Tanous BMCWEB_LOG_ERROR("SetPIDValues: Mapper Error"); 139373df0db0SJames Feist messages::internalError(self->asyncResp->res); 139473df0db0SJames Feist return; 139573df0db0SJames Feist } 139673df0db0SJames Feist 139773df0db0SJames Feist const std::string& path = subtree[0].first; 139873df0db0SJames Feist const std::string& owner = subtree[0].second[0].first; 1399fac6e53bSKrzysztof Grobelny sdbusplus::asio::getAllProperties( 1400fac6e53bSKrzysztof Grobelny *crow::connections::systemBus, owner, path, thermalModeIface, 14015e7e2dc5SEd Tanous [self, path, owner](const boost::system::error_code& ec2, 1402b9d36b47SEd Tanous const dbus::utility::DBusPropertiesMap& r) { 1403cb13a392SEd Tanous if (ec2) 140473df0db0SJames Feist { 140562598e31SEd Tanous BMCWEB_LOG_ERROR( 140662598e31SEd Tanous "SetPIDValues: Can't get thermalModeIface {}", path); 140773df0db0SJames Feist messages::internalError(self->asyncResp->res); 140873df0db0SJames Feist return; 140973df0db0SJames Feist } 1410271584abSEd Tanous const std::string* current = nullptr; 1411271584abSEd Tanous const std::vector<std::string>* supported = nullptr; 1412fac6e53bSKrzysztof Grobelny 1413fac6e53bSKrzysztof Grobelny const bool success = sdbusplus::unpackPropertiesNoThrow( 1414fac6e53bSKrzysztof Grobelny dbus_utils::UnpackErrorPrinter(), r, "Current", current, 1415fac6e53bSKrzysztof Grobelny "Supported", supported); 1416fac6e53bSKrzysztof Grobelny 1417fac6e53bSKrzysztof Grobelny if (!success) 141873df0db0SJames Feist { 1419002d39b4SEd Tanous messages::internalError(self->asyncResp->res); 142073df0db0SJames Feist return; 142173df0db0SJames Feist } 1422fac6e53bSKrzysztof Grobelny 142373df0db0SJames Feist if (current == nullptr || supported == nullptr) 142473df0db0SJames Feist { 142562598e31SEd Tanous BMCWEB_LOG_ERROR( 142662598e31SEd Tanous "SetPIDValues: thermal mode iface invalid {}", path); 142773df0db0SJames Feist messages::internalError(self->asyncResp->res); 142873df0db0SJames Feist return; 142973df0db0SJames Feist } 143073df0db0SJames Feist self->currentProfile = *current; 143173df0db0SJames Feist self->supportedProfiles = *supported; 143273df0db0SJames Feist self->profileConnection = owner; 143373df0db0SJames Feist self->profilePath = path; 1434fac6e53bSKrzysztof Grobelny }); 1435e99073f5SGeorge Liu }); 143673df0db0SJames Feist } 143724b2fe81SEd Tanous void pidSetDone() 143873df0db0SJames Feist { 143973df0db0SJames Feist if (asyncResp->res.result() != boost::beast::http::status::ok) 144073df0db0SJames Feist { 144173df0db0SJames Feist return; 14425b4aa86bSJames Feist } 14438d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> response = asyncResp; 144473df0db0SJames Feist if (profile) 144573df0db0SJames Feist { 14463544d2a7SEd Tanous if (std::ranges::find(supportedProfiles, *profile) == 14473544d2a7SEd Tanous supportedProfiles.end()) 144873df0db0SJames Feist { 144973df0db0SJames Feist messages::actionParameterUnknown(response->res, "Profile", 145073df0db0SJames Feist *profile); 145173df0db0SJames Feist return; 145273df0db0SJames Feist } 145373df0db0SJames Feist currentProfile = *profile; 14549ae226faSGeorge Liu sdbusplus::asio::setProperty( 14559ae226faSGeorge Liu *crow::connections::systemBus, profileConnection, profilePath, 14569ae226faSGeorge Liu thermalModeIface, "Current", *profile, 14575e7e2dc5SEd Tanous [response](const boost::system::error_code& ec) { 145873df0db0SJames Feist if (ec) 145973df0db0SJames Feist { 146062598e31SEd Tanous BMCWEB_LOG_ERROR("Error patching profile{}", ec); 146173df0db0SJames Feist messages::internalError(response->res); 146273df0db0SJames Feist } 14639ae226faSGeorge Liu }); 146473df0db0SJames Feist } 146573df0db0SJames Feist 146673df0db0SJames Feist for (auto& containerPair : configuration) 146773df0db0SJames Feist { 146873df0db0SJames Feist auto& container = containerPair.second; 146973df0db0SJames Feist if (!container) 147073df0db0SJames Feist { 147173df0db0SJames Feist continue; 147273df0db0SJames Feist } 14736ee7f774SJames Feist 147402cad96eSEd Tanous const std::string& type = containerPair.first; 147573df0db0SJames Feist 1476*9e9b6049SEd Tanous for (auto& [name, value] : *container) 147773df0db0SJames Feist { 1478cddbf3dfSPotin Lai std::string dbusObjName = name; 1479cddbf3dfSPotin Lai std::replace(dbusObjName.begin(), dbusObjName.end(), ' ', '_'); 148062598e31SEd Tanous BMCWEB_LOG_DEBUG("looking for {}", name); 14816ee7f774SJames Feist 14823544d2a7SEd Tanous auto pathItr = std::ranges::find_if( 14833544d2a7SEd Tanous managedObj, [&dbusObjName](const auto& obj) { 148418f8f608SEd Tanous return obj.first.parent_path() == dbusObjName; 148573df0db0SJames Feist }); 1486b9d36b47SEd Tanous dbus::utility::DBusPropertiesMap output; 148773df0db0SJames Feist 148873df0db0SJames Feist output.reserve(16); // The pid interface length 148973df0db0SJames Feist 149073df0db0SJames Feist // determines if we're patching entity-manager or 149173df0db0SJames Feist // creating a new object 149273df0db0SJames Feist bool createNewObject = (pathItr == managedObj.end()); 149362598e31SEd Tanous BMCWEB_LOG_DEBUG("Found = {}", !createNewObject); 14946ee7f774SJames Feist 149573df0db0SJames Feist std::string iface; 1496ea2b670dSEd Tanous if (!createNewObject) 1497ea2b670dSEd Tanous { 14988be2b5b6SPotin Lai bool findInterface = false; 1499ea2b670dSEd Tanous for (const auto& interface : pathItr->second) 1500ea2b670dSEd Tanous { 1501ea2b670dSEd Tanous if (interface.first == pidConfigurationIface) 1502ea2b670dSEd Tanous { 1503ea2b670dSEd Tanous if (type == "PidControllers" || 1504ea2b670dSEd Tanous type == "FanControllers") 150573df0db0SJames Feist { 150673df0db0SJames Feist iface = pidConfigurationIface; 15078be2b5b6SPotin Lai findInterface = true; 15088be2b5b6SPotin Lai break; 150973df0db0SJames Feist } 151073df0db0SJames Feist } 1511ea2b670dSEd Tanous else if (interface.first == pidZoneConfigurationIface) 151273df0db0SJames Feist { 1513ea2b670dSEd Tanous if (type == "FanZones") 151473df0db0SJames Feist { 1515da39350aSPavanKumarIntel iface = pidZoneConfigurationIface; 15168be2b5b6SPotin Lai findInterface = true; 15178be2b5b6SPotin Lai break; 151873df0db0SJames Feist } 151973df0db0SJames Feist } 1520ea2b670dSEd Tanous else if (interface.first == stepwiseConfigurationIface) 1521ea2b670dSEd Tanous { 1522ea2b670dSEd Tanous if (type == "StepwiseControllers") 152373df0db0SJames Feist { 152473df0db0SJames Feist iface = stepwiseConfigurationIface; 15258be2b5b6SPotin Lai findInterface = true; 15268be2b5b6SPotin Lai break; 15278be2b5b6SPotin Lai } 15288be2b5b6SPotin Lai } 15298be2b5b6SPotin Lai } 15308be2b5b6SPotin Lai 15318be2b5b6SPotin Lai // create new object if interface not found 15328be2b5b6SPotin Lai if (!findInterface) 15338be2b5b6SPotin Lai { 153473df0db0SJames Feist createNewObject = true; 153573df0db0SJames Feist } 1536ea2b670dSEd Tanous } 15376ee7f774SJames Feist 1538*9e9b6049SEd Tanous if (createNewObject && value == nullptr) 15396ee7f774SJames Feist { 15404e0453b1SGunnar Mills // can't delete a non-existent object 1541*9e9b6049SEd Tanous messages::propertyValueNotInList(response->res, value, 1542e2616cc5SEd Tanous name); 15436ee7f774SJames Feist continue; 15446ee7f774SJames Feist } 15456ee7f774SJames Feist 15466ee7f774SJames Feist std::string path; 15476ee7f774SJames Feist if (pathItr != managedObj.end()) 15486ee7f774SJames Feist { 15496ee7f774SJames Feist path = pathItr->first.str; 15506ee7f774SJames Feist } 15516ee7f774SJames Feist 155262598e31SEd Tanous BMCWEB_LOG_DEBUG("Create new = {}", createNewObject); 1553e69d9de2SJames Feist 1554e69d9de2SJames Feist // arbitrary limit to avoid attacks 1555e69d9de2SJames Feist constexpr const size_t controllerLimit = 500; 155614b0b8d5SJames Feist if (createNewObject && objectCount >= controllerLimit) 1557e69d9de2SJames Feist { 1558e69d9de2SJames Feist messages::resourceExhaustion(response->res, type); 1559e69d9de2SJames Feist continue; 1560e69d9de2SJames Feist } 1561a170f275SEd Tanous std::string escaped = name; 1562a170f275SEd Tanous std::replace(escaped.begin(), escaped.end(), '_', ' '); 1563a170f275SEd Tanous output.emplace_back("Name", escaped); 156473df0db0SJames Feist 156573df0db0SJames Feist std::string chassis; 156673df0db0SJames Feist CreatePIDRet ret = createPidInterface( 1567*9e9b6049SEd Tanous response, type, name, value, path, managedObj, 1568*9e9b6049SEd Tanous createNewObject, output, chassis, currentProfile); 156973df0db0SJames Feist if (ret == CreatePIDRet::fail) 157073df0db0SJames Feist { 157173df0db0SJames Feist return; 157273df0db0SJames Feist } 15733174e4dfSEd Tanous if (ret == CreatePIDRet::del) 157473df0db0SJames Feist { 157573df0db0SJames Feist continue; 157673df0db0SJames Feist } 157773df0db0SJames Feist 157873df0db0SJames Feist if (!createNewObject) 157973df0db0SJames Feist { 158073df0db0SJames Feist for (const auto& property : output) 158173df0db0SJames Feist { 15827a696974SPotin Lai crow::connections::systemBus->async_method_call( 158373df0db0SJames Feist [response, 158473df0db0SJames Feist propertyName{std::string(property.first)}]( 15855e7e2dc5SEd Tanous const boost::system::error_code& ec) { 158673df0db0SJames Feist if (ec) 158773df0db0SJames Feist { 158862598e31SEd Tanous BMCWEB_LOG_ERROR("Error patching {}: {}", 158962598e31SEd Tanous propertyName, ec); 159073df0db0SJames Feist messages::internalError(response->res); 159173df0db0SJames Feist return; 159273df0db0SJames Feist } 159373df0db0SJames Feist messages::success(response->res); 15947a696974SPotin Lai }, 15957a696974SPotin Lai "xyz.openbmc_project.EntityManager", path, 15967a696974SPotin Lai "org.freedesktop.DBus.Properties", "Set", iface, 15977a696974SPotin Lai property.first, property.second); 159873df0db0SJames Feist } 159973df0db0SJames Feist } 160073df0db0SJames Feist else 160173df0db0SJames Feist { 160273df0db0SJames Feist if (chassis.empty()) 160373df0db0SJames Feist { 160462598e31SEd Tanous BMCWEB_LOG_ERROR("Failed to get chassis from config"); 1605ace85d60SEd Tanous messages::internalError(response->res); 160673df0db0SJames Feist return; 160773df0db0SJames Feist } 160873df0db0SJames Feist 160973df0db0SJames Feist bool foundChassis = false; 161073df0db0SJames Feist for (const auto& obj : managedObj) 161173df0db0SJames Feist { 161218f8f608SEd Tanous if (obj.first.parent_path() == chassis) 161373df0db0SJames Feist { 161473df0db0SJames Feist chassis = obj.first.str; 161573df0db0SJames Feist foundChassis = true; 161673df0db0SJames Feist break; 161773df0db0SJames Feist } 161873df0db0SJames Feist } 161973df0db0SJames Feist if (!foundChassis) 162073df0db0SJames Feist { 162162598e31SEd Tanous BMCWEB_LOG_ERROR("Failed to find chassis on dbus"); 162273df0db0SJames Feist messages::resourceMissingAtURI( 1623ace85d60SEd Tanous response->res, 1624ef4c65b7SEd Tanous boost::urls::format("/redfish/v1/Chassis/{}", 1625ef4c65b7SEd Tanous chassis)); 162673df0db0SJames Feist return; 162773df0db0SJames Feist } 162873df0db0SJames Feist 162973df0db0SJames Feist crow::connections::systemBus->async_method_call( 16305e7e2dc5SEd Tanous [response](const boost::system::error_code& ec) { 163173df0db0SJames Feist if (ec) 163273df0db0SJames Feist { 163362598e31SEd Tanous BMCWEB_LOG_ERROR("Error Adding Pid Object {}", ec); 163473df0db0SJames Feist messages::internalError(response->res); 163573df0db0SJames Feist return; 163673df0db0SJames Feist } 163773df0db0SJames Feist messages::success(response->res); 163873df0db0SJames Feist }, 163973df0db0SJames Feist "xyz.openbmc_project.EntityManager", chassis, 164073df0db0SJames Feist "xyz.openbmc_project.AddObject", "AddObject", output); 164173df0db0SJames Feist } 164273df0db0SJames Feist } 164373df0db0SJames Feist } 164473df0db0SJames Feist } 164524b2fe81SEd Tanous 164624b2fe81SEd Tanous ~SetPIDValues() 164724b2fe81SEd Tanous { 164824b2fe81SEd Tanous try 164924b2fe81SEd Tanous { 165024b2fe81SEd Tanous pidSetDone(); 165124b2fe81SEd Tanous } 165224b2fe81SEd Tanous catch (...) 165324b2fe81SEd Tanous { 165462598e31SEd Tanous BMCWEB_LOG_CRITICAL("pidSetDone threw exception"); 165524b2fe81SEd Tanous } 165624b2fe81SEd Tanous } 165724b2fe81SEd Tanous 16588d1b46d7Szhanghch05 std::shared_ptr<bmcweb::AsyncResp> asyncResp; 1659*9e9b6049SEd Tanous std::vector<std::pair<std::string, std::optional<nlohmann::json::object_t>>> 166073df0db0SJames Feist configuration; 166173df0db0SJames Feist std::optional<std::string> profile; 166273df0db0SJames Feist dbus::utility::ManagedObjectType managedObj; 166373df0db0SJames Feist std::vector<std::string> supportedProfiles; 166473df0db0SJames Feist std::string currentProfile; 166573df0db0SJames Feist std::string profileConnection; 166673df0db0SJames Feist std::string profilePath; 166714b0b8d5SJames Feist size_t objectCount = 0; 166873df0db0SJames Feist }; 166973df0db0SJames Feist 1670071d8fdfSSunnySrivastava1984 /** 1671071d8fdfSSunnySrivastava1984 * @brief Retrieves BMC manager location data over DBus 1672071d8fdfSSunnySrivastava1984 * 1673ac106bf6SEd Tanous * @param[in] asyncResp Shared pointer for completing asynchronous calls 1674071d8fdfSSunnySrivastava1984 * @param[in] connectionName - service name 1675071d8fdfSSunnySrivastava1984 * @param[in] path - object path 1676071d8fdfSSunnySrivastava1984 * @return none 1677071d8fdfSSunnySrivastava1984 */ 1678ac106bf6SEd Tanous inline void getLocation(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1679071d8fdfSSunnySrivastava1984 const std::string& connectionName, 1680071d8fdfSSunnySrivastava1984 const std::string& path) 1681071d8fdfSSunnySrivastava1984 { 168262598e31SEd Tanous BMCWEB_LOG_DEBUG("Get BMC manager Location data."); 1683071d8fdfSSunnySrivastava1984 16841e1e598dSJonathan Doman sdbusplus::asio::getProperty<std::string>( 16851e1e598dSJonathan Doman *crow::connections::systemBus, connectionName, path, 16861e1e598dSJonathan Doman "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode", 1687ac106bf6SEd Tanous [asyncResp](const boost::system::error_code& ec, 16881e1e598dSJonathan Doman const std::string& property) { 1689071d8fdfSSunnySrivastava1984 if (ec) 1690071d8fdfSSunnySrivastava1984 { 169162598e31SEd Tanous BMCWEB_LOG_DEBUG("DBUS response error for " 169262598e31SEd Tanous "Location"); 1693ac106bf6SEd Tanous messages::internalError(asyncResp->res); 1694071d8fdfSSunnySrivastava1984 return; 1695071d8fdfSSunnySrivastava1984 } 1696071d8fdfSSunnySrivastava1984 1697ac106bf6SEd Tanous asyncResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] = 16981e1e598dSJonathan Doman property; 16991e1e598dSJonathan Doman }); 1700071d8fdfSSunnySrivastava1984 } 17017e860f15SJohn Edward Broadbent // avoid name collision systems.hpp 17027e860f15SJohn Edward Broadbent inline void 1703ac106bf6SEd Tanous managerGetLastResetTime(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 17044bf2b033SGunnar Mills { 170562598e31SEd Tanous BMCWEB_LOG_DEBUG("Getting Manager Last Reset Time"); 17064bf2b033SGunnar Mills 17071e1e598dSJonathan Doman sdbusplus::asio::getProperty<uint64_t>( 17081e1e598dSJonathan Doman *crow::connections::systemBus, "xyz.openbmc_project.State.BMC", 17091e1e598dSJonathan Doman "/xyz/openbmc_project/state/bmc0", "xyz.openbmc_project.State.BMC", 17101e1e598dSJonathan Doman "LastRebootTime", 1711ac106bf6SEd Tanous [asyncResp](const boost::system::error_code& ec, 17121e1e598dSJonathan Doman const uint64_t lastResetTime) { 17134bf2b033SGunnar Mills if (ec) 17144bf2b033SGunnar Mills { 171562598e31SEd Tanous BMCWEB_LOG_DEBUG("D-BUS response error {}", ec); 17164bf2b033SGunnar Mills return; 17174bf2b033SGunnar Mills } 17184bf2b033SGunnar Mills 17194bf2b033SGunnar Mills // LastRebootTime is epoch time, in milliseconds 17204bf2b033SGunnar Mills // https://github.com/openbmc/phosphor-dbus-interfaces/blob/7f9a128eb9296e926422ddc312c148b625890bb6/xyz/openbmc_project/State/BMC.interface.yaml#L19 17211e1e598dSJonathan Doman uint64_t lastResetTimeStamp = lastResetTime / 1000; 17224bf2b033SGunnar Mills 17234bf2b033SGunnar Mills // Convert to ISO 8601 standard 1724ac106bf6SEd Tanous asyncResp->res.jsonValue["LastResetTime"] = 17252b82937eSEd Tanous redfish::time_utils::getDateTimeUint(lastResetTimeStamp); 17261e1e598dSJonathan Doman }); 17274bf2b033SGunnar Mills } 17284bf2b033SGunnar Mills 17294bfefa74SGunnar Mills /** 17304bfefa74SGunnar Mills * @brief Set the running firmware image 17314bfefa74SGunnar Mills * 1732ac106bf6SEd Tanous * @param[i,o] asyncResp - Async response object 17334bfefa74SGunnar Mills * @param[i] runningFirmwareTarget - Image to make the running image 17344bfefa74SGunnar Mills * 17354bfefa74SGunnar Mills * @return void 17364bfefa74SGunnar Mills */ 17377e860f15SJohn Edward Broadbent inline void 1738ac106bf6SEd Tanous setActiveFirmwareImage(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1739f23b7296SEd Tanous const std::string& runningFirmwareTarget) 17404bfefa74SGunnar Mills { 17414bfefa74SGunnar Mills // Get the Id from /redfish/v1/UpdateService/FirmwareInventory/<Id> 1742f23b7296SEd Tanous std::string::size_type idPos = runningFirmwareTarget.rfind('/'); 17434bfefa74SGunnar Mills if (idPos == std::string::npos) 17444bfefa74SGunnar Mills { 1745ac106bf6SEd Tanous messages::propertyValueNotInList(asyncResp->res, runningFirmwareTarget, 17464bfefa74SGunnar Mills "@odata.id"); 174762598e31SEd Tanous BMCWEB_LOG_DEBUG("Can't parse firmware ID!"); 17484bfefa74SGunnar Mills return; 17494bfefa74SGunnar Mills } 17504bfefa74SGunnar Mills idPos++; 17514bfefa74SGunnar Mills if (idPos >= runningFirmwareTarget.size()) 17524bfefa74SGunnar Mills { 1753ac106bf6SEd Tanous messages::propertyValueNotInList(asyncResp->res, runningFirmwareTarget, 17544bfefa74SGunnar Mills "@odata.id"); 175562598e31SEd Tanous BMCWEB_LOG_DEBUG("Invalid firmware ID."); 17564bfefa74SGunnar Mills return; 17574bfefa74SGunnar Mills } 17584bfefa74SGunnar Mills std::string firmwareId = runningFirmwareTarget.substr(idPos); 17594bfefa74SGunnar Mills 17604bfefa74SGunnar Mills // Make sure the image is valid before setting priority 17615eb468daSGeorge Liu sdbusplus::message::object_path objPath("/xyz/openbmc_project/software"); 17625eb468daSGeorge Liu dbus::utility::getManagedObjects( 17635eb468daSGeorge Liu "xyz.openbmc_project.Software.BMC.Updater", objPath, 17645eb468daSGeorge Liu [asyncResp, firmwareId, runningFirmwareTarget]( 17655eb468daSGeorge Liu const boost::system::error_code& ec, 17665eb468daSGeorge Liu const dbus::utility::ManagedObjectType& subtree) { 17674bfefa74SGunnar Mills if (ec) 17684bfefa74SGunnar Mills { 176962598e31SEd Tanous BMCWEB_LOG_DEBUG("D-Bus response error getting objects."); 1770ac106bf6SEd Tanous messages::internalError(asyncResp->res); 17714bfefa74SGunnar Mills return; 17724bfefa74SGunnar Mills } 17734bfefa74SGunnar Mills 177426f6976fSEd Tanous if (subtree.empty()) 17754bfefa74SGunnar Mills { 177662598e31SEd Tanous BMCWEB_LOG_DEBUG("Can't find image!"); 1777ac106bf6SEd Tanous messages::internalError(asyncResp->res); 17784bfefa74SGunnar Mills return; 17794bfefa74SGunnar Mills } 17804bfefa74SGunnar Mills 17814bfefa74SGunnar Mills bool foundImage = false; 178202cad96eSEd Tanous for (const auto& object : subtree) 17834bfefa74SGunnar Mills { 17844bfefa74SGunnar Mills const std::string& path = 17854bfefa74SGunnar Mills static_cast<const std::string&>(object.first); 1786f23b7296SEd Tanous std::size_t idPos2 = path.rfind('/'); 17874bfefa74SGunnar Mills 17884bfefa74SGunnar Mills if (idPos2 == std::string::npos) 17894bfefa74SGunnar Mills { 17904bfefa74SGunnar Mills continue; 17914bfefa74SGunnar Mills } 17924bfefa74SGunnar Mills 17934bfefa74SGunnar Mills idPos2++; 17944bfefa74SGunnar Mills if (idPos2 >= path.size()) 17954bfefa74SGunnar Mills { 17964bfefa74SGunnar Mills continue; 17974bfefa74SGunnar Mills } 17984bfefa74SGunnar Mills 17994bfefa74SGunnar Mills if (path.substr(idPos2) == firmwareId) 18004bfefa74SGunnar Mills { 18014bfefa74SGunnar Mills foundImage = true; 18024bfefa74SGunnar Mills break; 18034bfefa74SGunnar Mills } 18044bfefa74SGunnar Mills } 18054bfefa74SGunnar Mills 18064bfefa74SGunnar Mills if (!foundImage) 18074bfefa74SGunnar Mills { 1808ac106bf6SEd Tanous messages::propertyValueNotInList( 1809ac106bf6SEd Tanous asyncResp->res, runningFirmwareTarget, "@odata.id"); 181062598e31SEd Tanous BMCWEB_LOG_DEBUG("Invalid firmware ID."); 18114bfefa74SGunnar Mills return; 18124bfefa74SGunnar Mills } 18134bfefa74SGunnar Mills 181462598e31SEd Tanous BMCWEB_LOG_DEBUG("Setting firmware version {} to priority 0.", 181562598e31SEd Tanous firmwareId); 18164bfefa74SGunnar Mills 18174bfefa74SGunnar Mills // Only support Immediate 18184bfefa74SGunnar Mills // An addition could be a Redfish Setting like 18194bfefa74SGunnar Mills // ActiveSoftwareImageApplyTime and support OnReset 18209ae226faSGeorge Liu sdbusplus::asio::setProperty( 18219ae226faSGeorge Liu *crow::connections::systemBus, 18229ae226faSGeorge Liu "xyz.openbmc_project.Software.BMC.Updater", 18239ae226faSGeorge Liu "/xyz/openbmc_project/software/" + firmwareId, 18249ae226faSGeorge Liu "xyz.openbmc_project.Software.RedundancyPriority", "Priority", 18259ae226faSGeorge Liu static_cast<uint8_t>(0), 1826ac106bf6SEd Tanous [asyncResp](const boost::system::error_code& ec2) { 18278a592810SEd Tanous if (ec2) 18284bfefa74SGunnar Mills { 182962598e31SEd Tanous BMCWEB_LOG_DEBUG("D-Bus response error setting."); 1830ac106bf6SEd Tanous messages::internalError(asyncResp->res); 18314bfefa74SGunnar Mills return; 18324bfefa74SGunnar Mills } 1833ac106bf6SEd Tanous doBMCGracefulRestart(asyncResp); 18349ae226faSGeorge Liu }); 18355eb468daSGeorge Liu }); 18364bfefa74SGunnar Mills } 18374bfefa74SGunnar Mills 1838c51afd54SEd Tanous inline void 1839c51afd54SEd Tanous afterSetDateTime(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1840c51afd54SEd Tanous const boost::system::error_code& ec, 1841c51afd54SEd Tanous const sdbusplus::message_t& msg) 1842c51afd54SEd Tanous { 1843c51afd54SEd Tanous if (ec) 1844c51afd54SEd Tanous { 1845c51afd54SEd Tanous BMCWEB_LOG_DEBUG("Failed to set elapsed time. DBUS response error {}", 1846c51afd54SEd Tanous ec); 1847c51afd54SEd Tanous const sd_bus_error* dbusError = msg.get_error(); 1848c51afd54SEd Tanous if (dbusError != nullptr) 1849c51afd54SEd Tanous { 1850c51afd54SEd Tanous std::string_view errorName(dbusError->name); 1851c51afd54SEd Tanous if (errorName == 1852c51afd54SEd Tanous "org.freedesktop.timedate1.AutomaticTimeSyncEnabled") 1853c51afd54SEd Tanous { 1854c51afd54SEd Tanous BMCWEB_LOG_DEBUG("Setting conflict"); 1855c51afd54SEd Tanous messages::propertyValueConflict( 1856c51afd54SEd Tanous asyncResp->res, "DateTime", 1857c51afd54SEd Tanous "Managers/NetworkProtocol/NTPProcotolEnabled"); 1858c51afd54SEd Tanous return; 1859c51afd54SEd Tanous } 1860c51afd54SEd Tanous } 1861c51afd54SEd Tanous messages::internalError(asyncResp->res); 1862c51afd54SEd Tanous return; 1863c51afd54SEd Tanous } 1864c51afd54SEd Tanous asyncResp->res.result(boost::beast::http::status::no_content); 1865c51afd54SEd Tanous } 1866c51afd54SEd Tanous 1867c51afd54SEd Tanous inline void setDateTime(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1868c51afd54SEd Tanous const std::string& datetime) 1869af5d6058SSantosh Puranik { 187062598e31SEd Tanous BMCWEB_LOG_DEBUG("Set date time: {}", datetime); 1871af5d6058SSantosh Puranik 1872c2e32007SEd Tanous std::optional<redfish::time_utils::usSinceEpoch> us = 1873c2e32007SEd Tanous redfish::time_utils::dateStringToEpoch(datetime); 1874c2e32007SEd Tanous if (!us) 1875af5d6058SSantosh Puranik { 1876ac106bf6SEd Tanous messages::propertyValueFormatError(asyncResp->res, datetime, 1877ac106bf6SEd Tanous "DateTime"); 1878c2e32007SEd Tanous return; 1879c2e32007SEd Tanous } 1880c51afd54SEd Tanous // Set the absolute datetime 1881c51afd54SEd Tanous bool relative = false; 1882c51afd54SEd Tanous bool interactive = false; 1883c51afd54SEd Tanous crow::connections::systemBus->async_method_call( 1884c51afd54SEd Tanous [asyncResp](const boost::system::error_code& ec, 1885c51afd54SEd Tanous const sdbusplus::message_t& msg) { 1886c51afd54SEd Tanous afterSetDateTime(asyncResp, ec, msg); 1887c51afd54SEd Tanous }, 1888c51afd54SEd Tanous "org.freedesktop.timedate1", "/org/freedesktop/timedate1", 1889c51afd54SEd Tanous "org.freedesktop.timedate1", "SetTime", us->count(), relative, 1890c51afd54SEd Tanous interactive); 189183ff9ab6SJames Feist } 18929c310685SBorawski.Lukasz 189375815e5cSEd Tanous inline void 189475815e5cSEd Tanous checkForQuiesced(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 189575815e5cSEd Tanous { 189675815e5cSEd Tanous sdbusplus::asio::getProperty<std::string>( 189775815e5cSEd Tanous *crow::connections::systemBus, "org.freedesktop.systemd1", 189875815e5cSEd Tanous "/org/freedesktop/systemd1/unit/obmc-bmc-service-quiesce@0.target", 189975815e5cSEd Tanous "org.freedesktop.systemd1.Unit", "ActiveState", 190075815e5cSEd Tanous [asyncResp](const boost::system::error_code& ec, 190175815e5cSEd Tanous const std::string& val) { 190275815e5cSEd Tanous if (!ec) 190375815e5cSEd Tanous { 190475815e5cSEd Tanous if (val == "active") 190575815e5cSEd Tanous { 190675815e5cSEd Tanous asyncResp->res.jsonValue["Status"]["Health"] = "Critical"; 190775815e5cSEd Tanous asyncResp->res.jsonValue["Status"]["State"] = "Quiesced"; 190875815e5cSEd Tanous return; 190975815e5cSEd Tanous } 191075815e5cSEd Tanous } 191175815e5cSEd Tanous asyncResp->res.jsonValue["Status"]["Health"] = "OK"; 191275815e5cSEd Tanous asyncResp->res.jsonValue["Status"]["State"] = "Enabled"; 191375815e5cSEd Tanous }); 191475815e5cSEd Tanous } 191575815e5cSEd Tanous 19167e860f15SJohn Edward Broadbent inline void requestRoutesManager(App& app) 19177e860f15SJohn Edward Broadbent { 19187e860f15SJohn Edward Broadbent std::string uuid = persistent_data::getConfig().systemUuid; 19199c310685SBorawski.Lukasz 19207e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/") 1921ed398213SEd Tanous .privileges(redfish::privileges::getManager) 1922002d39b4SEd Tanous .methods(boost::beast::http::verb::get)( 1923002d39b4SEd Tanous [&app, uuid](const crow::Request& req, 192445ca1b86SEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 19253ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 192645ca1b86SEd Tanous { 192745ca1b86SEd Tanous return; 192845ca1b86SEd Tanous } 19297e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc"; 1930a51fc2d2SSui Chen asyncResp->res.jsonValue["@odata.type"] = "#Manager.v1_14_0.Manager"; 19317e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Id"] = "bmc"; 19327e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Name"] = "OpenBmc Manager"; 19337e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Description"] = 19347e860f15SJohn Edward Broadbent "Baseboard Management Controller"; 19357e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["PowerState"] = "On"; 19361476687dSEd Tanous 19377e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["ManagerType"] = "BMC"; 19387e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["UUID"] = systemd_utils::getUuid(); 19397e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["ServiceEntryPointUUID"] = uuid; 1940002d39b4SEd Tanous asyncResp->res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model 19417e860f15SJohn Edward Broadbent 19421476687dSEd Tanous asyncResp->res.jsonValue["LogServices"]["@odata.id"] = 19431476687dSEd Tanous "/redfish/v1/Managers/bmc/LogServices"; 19441476687dSEd Tanous asyncResp->res.jsonValue["NetworkProtocol"]["@odata.id"] = 19451476687dSEd Tanous "/redfish/v1/Managers/bmc/NetworkProtocol"; 19461476687dSEd Tanous asyncResp->res.jsonValue["EthernetInterfaces"]["@odata.id"] = 19471476687dSEd Tanous "/redfish/v1/Managers/bmc/EthernetInterfaces"; 19487e860f15SJohn Edward Broadbent 19497e860f15SJohn Edward Broadbent #ifdef BMCWEB_ENABLE_VM_NBDPROXY 19501476687dSEd Tanous asyncResp->res.jsonValue["VirtualMedia"]["@odata.id"] = 19511476687dSEd Tanous "/redfish/v1/Managers/bmc/VirtualMedia"; 19527e860f15SJohn Edward Broadbent #endif // BMCWEB_ENABLE_VM_NBDPROXY 19537e860f15SJohn Edward Broadbent 19547e860f15SJohn Edward Broadbent // default oem data 19557e860f15SJohn Edward Broadbent nlohmann::json& oem = asyncResp->res.jsonValue["Oem"]; 19567e860f15SJohn Edward Broadbent nlohmann::json& oemOpenbmc = oem["OpenBmc"]; 19577e860f15SJohn Edward Broadbent oem["@odata.type"] = "#OemManager.Oem"; 19587e860f15SJohn Edward Broadbent oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem"; 19597e860f15SJohn Edward Broadbent oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc"; 19607e860f15SJohn Edward Broadbent oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc"; 19611476687dSEd Tanous 19621476687dSEd Tanous nlohmann::json::object_t certificates; 19631476687dSEd Tanous certificates["@odata.id"] = 19641476687dSEd Tanous "/redfish/v1/Managers/bmc/Truststore/Certificates"; 19651476687dSEd Tanous oemOpenbmc["Certificates"] = std::move(certificates); 19667e860f15SJohn Edward Broadbent 19677e860f15SJohn Edward Broadbent // Manager.Reset (an action) can be many values, OpenBMC only 19687e860f15SJohn Edward Broadbent // supports BMC reboot. 19697e860f15SJohn Edward Broadbent nlohmann::json& managerReset = 19707e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Actions"]["#Manager.Reset"]; 19717e860f15SJohn Edward Broadbent managerReset["target"] = 19727e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/Actions/Manager.Reset"; 19737e860f15SJohn Edward Broadbent managerReset["@Redfish.ActionInfo"] = 19747e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/ResetActionInfo"; 19757e860f15SJohn Edward Broadbent 19767e860f15SJohn Edward Broadbent // ResetToDefaults (Factory Reset) has values like 19777e860f15SJohn Edward Broadbent // PreserveNetworkAndUsers and PreserveNetwork that aren't supported 19787e860f15SJohn Edward Broadbent // on OpenBMC 19797e860f15SJohn Edward Broadbent nlohmann::json& resetToDefaults = 19807e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["Actions"]["#Manager.ResetToDefaults"]; 19817e860f15SJohn Edward Broadbent resetToDefaults["target"] = 19827e860f15SJohn Edward Broadbent "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults"; 1983613dabeaSEd Tanous resetToDefaults["ResetType@Redfish.AllowableValues"] = 1984613dabeaSEd Tanous nlohmann::json::array_t({"ResetAll"}); 19857e860f15SJohn Edward Broadbent 19867c8c4058STejas Patil std::pair<std::string, std::string> redfishDateTimeOffset = 19872b82937eSEd Tanous redfish::time_utils::getDateTimeOffsetNow(); 19887c8c4058STejas Patil 19897c8c4058STejas Patil asyncResp->res.jsonValue["DateTime"] = redfishDateTimeOffset.first; 19907c8c4058STejas Patil asyncResp->res.jsonValue["DateTimeLocalOffset"] = 19917c8c4058STejas Patil redfishDateTimeOffset.second; 19927e860f15SJohn Edward Broadbent 19930e8ac5e7SGunnar Mills // TODO (Gunnar): Remove these one day since moved to ComputerSystem 19940e8ac5e7SGunnar Mills // Still used by OCP profiles 19950e8ac5e7SGunnar Mills // https://github.com/opencomputeproject/OCP-Profiles/issues/23 19967e860f15SJohn Edward Broadbent // Fill in SerialConsole info 19977e860f15SJohn Edward Broadbent asyncResp->res.jsonValue["SerialConsole"]["ServiceEnabled"] = true; 1998002d39b4SEd Tanous asyncResp->res.jsonValue["SerialConsole"]["MaxConcurrentSessions"] = 15; 1999613dabeaSEd Tanous asyncResp->res.jsonValue["SerialConsole"]["ConnectTypesSupported"] = 2000613dabeaSEd Tanous nlohmann::json::array_t({"IPMI", "SSH"}); 20017e860f15SJohn Edward Broadbent #ifdef BMCWEB_ENABLE_KVM 20027e860f15SJohn Edward Broadbent // Fill in GraphicalConsole info 2003002d39b4SEd Tanous asyncResp->res.jsonValue["GraphicalConsole"]["ServiceEnabled"] = true; 2004002d39b4SEd Tanous asyncResp->res.jsonValue["GraphicalConsole"]["MaxConcurrentSessions"] = 2005002d39b4SEd Tanous 4; 2006613dabeaSEd Tanous asyncResp->res.jsonValue["GraphicalConsole"]["ConnectTypesSupported"] = 2007613dabeaSEd Tanous nlohmann::json::array_t({"KVMIP"}); 20087e860f15SJohn Edward Broadbent #endif // BMCWEB_ENABLE_KVM 20097f3e84a1SEd Tanous if constexpr (!bmcwebEnableMultiHost) 20107f3e84a1SEd Tanous { 20117f3e84a1SEd Tanous asyncResp->res.jsonValue["Links"]["ManagerForServers@odata.count"] = 20127f3e84a1SEd Tanous 1; 20131476687dSEd Tanous 20141476687dSEd Tanous nlohmann::json::array_t managerForServers; 20151476687dSEd Tanous nlohmann::json::object_t manager; 20161476687dSEd Tanous manager["@odata.id"] = "/redfish/v1/Systems/system"; 2017ad539545SPatrick Williams managerForServers.emplace_back(std::move(manager)); 20181476687dSEd Tanous 20191476687dSEd Tanous asyncResp->res.jsonValue["Links"]["ManagerForServers"] = 20201476687dSEd Tanous std::move(managerForServers); 20217f3e84a1SEd Tanous } 202213451e39SWilly Tu if constexpr (bmcwebEnableHealthPopulate) 202313451e39SWilly Tu { 20247e860f15SJohn Edward Broadbent auto health = std::make_shared<HealthPopulate>(asyncResp); 20257e860f15SJohn Edward Broadbent health->isManagersHealth = true; 20267e860f15SJohn Edward Broadbent health->populate(); 202713451e39SWilly Tu } 20287e860f15SJohn Edward Broadbent 2029eee0013eSWilly Tu sw_util::populateSoftwareInformation(asyncResp, sw_util::bmcPurpose, 20307e860f15SJohn Edward Broadbent "FirmwareVersion", true); 20317e860f15SJohn Edward Broadbent 20327e860f15SJohn Edward Broadbent managerGetLastResetTime(asyncResp); 20337e860f15SJohn Edward Broadbent 2034a51fc2d2SSui Chen // ManagerDiagnosticData is added for all BMCs. 2035a51fc2d2SSui Chen nlohmann::json& managerDiagnosticData = 2036a51fc2d2SSui Chen asyncResp->res.jsonValue["ManagerDiagnosticData"]; 2037a51fc2d2SSui Chen managerDiagnosticData["@odata.id"] = 2038a51fc2d2SSui Chen "/redfish/v1/Managers/bmc/ManagerDiagnosticData"; 2039a51fc2d2SSui Chen 204054dce7f5SGunnar Mills #ifdef BMCWEB_ENABLE_REDFISH_OEM_MANAGER_FAN_DATA 20417e860f15SJohn Edward Broadbent auto pids = std::make_shared<GetPIDValues>(asyncResp); 20427e860f15SJohn Edward Broadbent pids->run(); 204354dce7f5SGunnar Mills #endif 20447e860f15SJohn Edward Broadbent 2045002d39b4SEd Tanous getMainChassisId(asyncResp, 2046002d39b4SEd Tanous [](const std::string& chassisId, 2047002d39b4SEd Tanous const std::shared_ptr<bmcweb::AsyncResp>& aRsp) { 2048002d39b4SEd Tanous aRsp->res.jsonValue["Links"]["ManagerForChassis@odata.count"] = 1; 20491476687dSEd Tanous nlohmann::json::array_t managerForChassis; 20508a592810SEd Tanous nlohmann::json::object_t managerObj; 2051ef4c65b7SEd Tanous boost::urls::url chassiUrl = 2052ef4c65b7SEd Tanous boost::urls::format("/redfish/v1/Chassis/{}", chassisId); 2053eddfc437SWilly Tu managerObj["@odata.id"] = chassiUrl; 2054ad539545SPatrick Williams managerForChassis.emplace_back(std::move(managerObj)); 20551476687dSEd Tanous aRsp->res.jsonValue["Links"]["ManagerForChassis"] = 20561476687dSEd Tanous std::move(managerForChassis); 20571476687dSEd Tanous aRsp->res.jsonValue["Links"]["ManagerInChassis"]["@odata.id"] = 2058eddfc437SWilly Tu chassiUrl; 20597e860f15SJohn Edward Broadbent }); 20607e860f15SJohn Edward Broadbent 20611e1e598dSJonathan Doman sdbusplus::asio::getProperty<double>( 20621e1e598dSJonathan Doman *crow::connections::systemBus, "org.freedesktop.systemd1", 2063002d39b4SEd Tanous "/org/freedesktop/systemd1", "org.freedesktop.systemd1.Manager", 2064002d39b4SEd Tanous "Progress", 206575815e5cSEd Tanous [asyncResp](const boost::system::error_code& ec, double val) { 20667e860f15SJohn Edward Broadbent if (ec) 20671abe55efSEd Tanous { 206862598e31SEd Tanous BMCWEB_LOG_ERROR("Error while getting progress"); 20697e860f15SJohn Edward Broadbent messages::internalError(asyncResp->res); 20707e860f15SJohn Edward Broadbent return; 20717e860f15SJohn Edward Broadbent } 20721e1e598dSJonathan Doman if (val < 1.0) 20737e860f15SJohn Edward Broadbent { 207475815e5cSEd Tanous asyncResp->res.jsonValue["Status"]["Health"] = "OK"; 2075002d39b4SEd Tanous asyncResp->res.jsonValue["Status"]["State"] = "Starting"; 207675815e5cSEd Tanous return; 20777e860f15SJohn Edward Broadbent } 207875815e5cSEd Tanous checkForQuiesced(asyncResp); 20791e1e598dSJonathan Doman }); 20809c310685SBorawski.Lukasz 2081e99073f5SGeorge Liu constexpr std::array<std::string_view, 1> interfaces = { 2082e99073f5SGeorge Liu "xyz.openbmc_project.Inventory.Item.Bmc"}; 2083e99073f5SGeorge Liu dbus::utility::getSubTree( 2084e99073f5SGeorge Liu "/xyz/openbmc_project/inventory", 0, interfaces, 20857e860f15SJohn Edward Broadbent [asyncResp]( 2086e99073f5SGeorge Liu const boost::system::error_code& ec, 2087b9d36b47SEd Tanous const dbus::utility::MapperGetSubTreeResponse& subtree) { 20887e860f15SJohn Edward Broadbent if (ec) 20891abe55efSEd Tanous { 209062598e31SEd Tanous BMCWEB_LOG_DEBUG("D-Bus response error on GetSubTree {}", ec); 20917e860f15SJohn Edward Broadbent return; 20927e860f15SJohn Edward Broadbent } 209326f6976fSEd Tanous if (subtree.empty()) 20947e860f15SJohn Edward Broadbent { 209562598e31SEd Tanous BMCWEB_LOG_DEBUG("Can't find bmc D-Bus object!"); 20967e860f15SJohn Edward Broadbent return; 20977e860f15SJohn Edward Broadbent } 20987e860f15SJohn Edward Broadbent // Assume only 1 bmc D-Bus object 20997e860f15SJohn Edward Broadbent // Throw an error if there is more than 1 21007e860f15SJohn Edward Broadbent if (subtree.size() > 1) 21017e860f15SJohn Edward Broadbent { 210262598e31SEd Tanous BMCWEB_LOG_DEBUG("Found more than 1 bmc D-Bus object!"); 21037e860f15SJohn Edward Broadbent messages::internalError(asyncResp->res); 21047e860f15SJohn Edward Broadbent return; 21057e860f15SJohn Edward Broadbent } 21067e860f15SJohn Edward Broadbent 2107002d39b4SEd Tanous if (subtree[0].first.empty() || subtree[0].second.size() != 1) 21087e860f15SJohn Edward Broadbent { 210962598e31SEd Tanous BMCWEB_LOG_DEBUG("Error getting bmc D-Bus object!"); 21107e860f15SJohn Edward Broadbent messages::internalError(asyncResp->res); 21117e860f15SJohn Edward Broadbent return; 21127e860f15SJohn Edward Broadbent } 21137e860f15SJohn Edward Broadbent 21147e860f15SJohn Edward Broadbent const std::string& path = subtree[0].first; 2115002d39b4SEd Tanous const std::string& connectionName = subtree[0].second[0].first; 21167e860f15SJohn Edward Broadbent 2117002d39b4SEd Tanous for (const auto& interfaceName : subtree[0].second[0].second) 21187e860f15SJohn Edward Broadbent { 21197e860f15SJohn Edward Broadbent if (interfaceName == 21207e860f15SJohn Edward Broadbent "xyz.openbmc_project.Inventory.Decorator.Asset") 21217e860f15SJohn Edward Broadbent { 2122fac6e53bSKrzysztof Grobelny sdbusplus::asio::getAllProperties( 2123fac6e53bSKrzysztof Grobelny *crow::connections::systemBus, connectionName, path, 2124fac6e53bSKrzysztof Grobelny "xyz.openbmc_project.Inventory.Decorator.Asset", 21255e7e2dc5SEd Tanous [asyncResp](const boost::system::error_code& ec2, 2126b9d36b47SEd Tanous const dbus::utility::DBusPropertiesMap& 21277e860f15SJohn Edward Broadbent propertiesList) { 21288a592810SEd Tanous if (ec2) 21297e860f15SJohn Edward Broadbent { 213062598e31SEd Tanous BMCWEB_LOG_DEBUG("Can't get bmc asset!"); 21317e860f15SJohn Edward Broadbent return; 21327e860f15SJohn Edward Broadbent } 21337e860f15SJohn Edward Broadbent 2134fac6e53bSKrzysztof Grobelny const std::string* partNumber = nullptr; 2135fac6e53bSKrzysztof Grobelny const std::string* serialNumber = nullptr; 2136fac6e53bSKrzysztof Grobelny const std::string* manufacturer = nullptr; 2137fac6e53bSKrzysztof Grobelny const std::string* model = nullptr; 2138fac6e53bSKrzysztof Grobelny const std::string* sparePartNumber = nullptr; 2139fac6e53bSKrzysztof Grobelny 2140fac6e53bSKrzysztof Grobelny const bool success = sdbusplus::unpackPropertiesNoThrow( 2141fac6e53bSKrzysztof Grobelny dbus_utils::UnpackErrorPrinter(), propertiesList, 2142fac6e53bSKrzysztof Grobelny "PartNumber", partNumber, "SerialNumber", 2143fac6e53bSKrzysztof Grobelny serialNumber, "Manufacturer", manufacturer, "Model", 2144fac6e53bSKrzysztof Grobelny model, "SparePartNumber", sparePartNumber); 2145fac6e53bSKrzysztof Grobelny 2146fac6e53bSKrzysztof Grobelny if (!success) 21477e860f15SJohn Edward Broadbent { 2148002d39b4SEd Tanous messages::internalError(asyncResp->res); 21497e860f15SJohn Edward Broadbent return; 21507e860f15SJohn Edward Broadbent } 2151fac6e53bSKrzysztof Grobelny 2152fac6e53bSKrzysztof Grobelny if (partNumber != nullptr) 2153fac6e53bSKrzysztof Grobelny { 2154fac6e53bSKrzysztof Grobelny asyncResp->res.jsonValue["PartNumber"] = 2155fac6e53bSKrzysztof Grobelny *partNumber; 21567e860f15SJohn Edward Broadbent } 2157fac6e53bSKrzysztof Grobelny 2158fac6e53bSKrzysztof Grobelny if (serialNumber != nullptr) 2159fac6e53bSKrzysztof Grobelny { 2160fac6e53bSKrzysztof Grobelny asyncResp->res.jsonValue["SerialNumber"] = 2161fac6e53bSKrzysztof Grobelny *serialNumber; 21627e860f15SJohn Edward Broadbent } 2163fac6e53bSKrzysztof Grobelny 2164fac6e53bSKrzysztof Grobelny if (manufacturer != nullptr) 2165fac6e53bSKrzysztof Grobelny { 2166fac6e53bSKrzysztof Grobelny asyncResp->res.jsonValue["Manufacturer"] = 2167fac6e53bSKrzysztof Grobelny *manufacturer; 2168fac6e53bSKrzysztof Grobelny } 2169fac6e53bSKrzysztof Grobelny 2170fac6e53bSKrzysztof Grobelny if (model != nullptr) 2171fac6e53bSKrzysztof Grobelny { 2172fac6e53bSKrzysztof Grobelny asyncResp->res.jsonValue["Model"] = *model; 2173fac6e53bSKrzysztof Grobelny } 2174fac6e53bSKrzysztof Grobelny 2175fac6e53bSKrzysztof Grobelny if (sparePartNumber != nullptr) 2176fac6e53bSKrzysztof Grobelny { 2177fac6e53bSKrzysztof Grobelny asyncResp->res.jsonValue["SparePartNumber"] = 2178fac6e53bSKrzysztof Grobelny *sparePartNumber; 2179fac6e53bSKrzysztof Grobelny } 2180fac6e53bSKrzysztof Grobelny }); 21817e860f15SJohn Edward Broadbent } 2182002d39b4SEd Tanous else if (interfaceName == 21830fda0f12SGeorge Liu "xyz.openbmc_project.Inventory.Decorator.LocationCode") 21847e860f15SJohn Edward Broadbent { 21857e860f15SJohn Edward Broadbent getLocation(asyncResp, connectionName, path); 21867e860f15SJohn Edward Broadbent } 21877e860f15SJohn Edward Broadbent } 2188e99073f5SGeorge Liu }); 21897e860f15SJohn Edward Broadbent }); 21907e860f15SJohn Edward Broadbent 21917e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/") 2192ed398213SEd Tanous .privileges(redfish::privileges::patchManager) 219345ca1b86SEd Tanous .methods(boost::beast::http::verb::patch)( 219445ca1b86SEd Tanous [&app](const crow::Request& req, 21957e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 21963ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 219745ca1b86SEd Tanous { 219845ca1b86SEd Tanous return; 219945ca1b86SEd Tanous } 2200*9e9b6049SEd Tanous std::optional<std::string> activeSoftwareImageOdataId; 22017e860f15SJohn Edward Broadbent std::optional<std::string> datetime; 2202*9e9b6049SEd Tanous std::optional<nlohmann::json::object_t> pidControllers; 2203*9e9b6049SEd Tanous std::optional<nlohmann::json::object_t> fanControllers; 2204*9e9b6049SEd Tanous std::optional<nlohmann::json::object_t> fanZones; 2205*9e9b6049SEd Tanous std::optional<nlohmann::json::object_t> stepwiseControllers; 2206*9e9b6049SEd Tanous std::optional<std::string> profile; 22077e860f15SJohn Edward Broadbent 2208*9e9b6049SEd Tanous // clang-format off 2209*9e9b6049SEd Tanous if (!json_util::readJsonPatch(req, asyncResp->res, 2210*9e9b6049SEd Tanous "DateTime", datetime, 2211*9e9b6049SEd Tanous "Links/ActiveSoftwareImage/@odata.id", activeSoftwareImageOdataId, 2212*9e9b6049SEd Tanous "Oem/OpenBmc/Fan/FanControllers", fanControllers, 2213*9e9b6049SEd Tanous "Oem/OpenBmc/Fan/FanZones", fanZones, 2214*9e9b6049SEd Tanous "Oem/OpenBmc/Fan/PidControllers", pidControllers, 2215*9e9b6049SEd Tanous "Oem/OpenBmc/Fan/Profile", profile, 2216*9e9b6049SEd Tanous "Oem/OpenBmc/Fan/StepwiseControllers", stepwiseControllers 2217*9e9b6049SEd Tanous )) 22187e860f15SJohn Edward Broadbent { 22197e860f15SJohn Edward Broadbent return; 22207e860f15SJohn Edward Broadbent } 2221*9e9b6049SEd Tanous // clang-format on 22227e860f15SJohn Edward Broadbent 2223*9e9b6049SEd Tanous if (pidControllers || fanControllers || fanZones || 2224*9e9b6049SEd Tanous stepwiseControllers || profile) 22257e860f15SJohn Edward Broadbent { 222654dce7f5SGunnar Mills #ifdef BMCWEB_ENABLE_REDFISH_OEM_MANAGER_FAN_DATA 2227*9e9b6049SEd Tanous std::vector< 2228*9e9b6049SEd Tanous std::pair<std::string, std::optional<nlohmann::json::object_t>>> 2229*9e9b6049SEd Tanous configuration; 2230*9e9b6049SEd Tanous if (pidControllers) 22317e860f15SJohn Edward Broadbent { 2232*9e9b6049SEd Tanous configuration.emplace_back("PidControllers", 2233*9e9b6049SEd Tanous std::move(pidControllers)); 22347e860f15SJohn Edward Broadbent } 2235*9e9b6049SEd Tanous if (fanControllers) 22367e860f15SJohn Edward Broadbent { 2237*9e9b6049SEd Tanous configuration.emplace_back("FanControllers", 2238*9e9b6049SEd Tanous std::move(fanControllers)); 22397e860f15SJohn Edward Broadbent } 2240*9e9b6049SEd Tanous if (fanZones) 22417e860f15SJohn Edward Broadbent { 2242*9e9b6049SEd Tanous configuration.emplace_back("FanZones", std::move(fanZones)); 2243*9e9b6049SEd Tanous } 2244*9e9b6049SEd Tanous if (stepwiseControllers) 2245*9e9b6049SEd Tanous { 2246*9e9b6049SEd Tanous configuration.emplace_back("StepwiseControllers", 2247*9e9b6049SEd Tanous std::move(stepwiseControllers)); 2248*9e9b6049SEd Tanous } 2249*9e9b6049SEd Tanous auto pid = std::make_shared<SetPIDValues>( 2250*9e9b6049SEd Tanous asyncResp, std::move(configuration), profile); 22517e860f15SJohn Edward Broadbent pid->run(); 225254dce7f5SGunnar Mills #else 225354dce7f5SGunnar Mills messages::propertyUnknown(asyncResp->res, "Oem"); 225454dce7f5SGunnar Mills return; 225554dce7f5SGunnar Mills #endif 22567e860f15SJohn Edward Broadbent } 2257*9e9b6049SEd Tanous 2258*9e9b6049SEd Tanous if (activeSoftwareImageOdataId) 22597e860f15SJohn Edward Broadbent { 2260*9e9b6049SEd Tanous setActiveFirmwareImage(asyncResp, *activeSoftwareImageOdataId); 22617e860f15SJohn Edward Broadbent } 22627e860f15SJohn Edward Broadbent 22637e860f15SJohn Edward Broadbent if (datetime) 22647e860f15SJohn Edward Broadbent { 2265c51afd54SEd Tanous setDateTime(asyncResp, *datetime); 22667e860f15SJohn Edward Broadbent } 22677e860f15SJohn Edward Broadbent }); 22687e860f15SJohn Edward Broadbent } 22697e860f15SJohn Edward Broadbent 22707e860f15SJohn Edward Broadbent inline void requestRoutesManagerCollection(App& app) 22717e860f15SJohn Edward Broadbent { 22727e860f15SJohn Edward Broadbent BMCWEB_ROUTE(app, "/redfish/v1/Managers/") 2273ed398213SEd Tanous .privileges(redfish::privileges::getManagerCollection) 22747e860f15SJohn Edward Broadbent .methods(boost::beast::http::verb::get)( 227545ca1b86SEd Tanous [&app](const crow::Request& req, 22767e860f15SJohn Edward Broadbent const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 22773ba00073SCarson Labrado if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 227845ca1b86SEd Tanous { 227945ca1b86SEd Tanous return; 228045ca1b86SEd Tanous } 228183ff9ab6SJames Feist // Collections don't include the static data added by SubRoute 228283ff9ab6SJames Feist // because it has a duplicate entry for members 22838d1b46d7Szhanghch05 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers"; 22848d1b46d7Szhanghch05 asyncResp->res.jsonValue["@odata.type"] = 22858d1b46d7Szhanghch05 "#ManagerCollection.ManagerCollection"; 22868d1b46d7Szhanghch05 asyncResp->res.jsonValue["Name"] = "Manager Collection"; 22878d1b46d7Szhanghch05 asyncResp->res.jsonValue["Members@odata.count"] = 1; 22881476687dSEd Tanous nlohmann::json::array_t members; 22891476687dSEd Tanous nlohmann::json& bmc = members.emplace_back(); 22901476687dSEd Tanous bmc["@odata.id"] = "/redfish/v1/Managers/bmc"; 22911476687dSEd Tanous asyncResp->res.jsonValue["Members"] = std::move(members); 22927e860f15SJohn Edward Broadbent }); 22939c310685SBorawski.Lukasz } 22949c310685SBorawski.Lukasz } // namespace redfish 2295