xref: /openbmc/bmcweb/features/redfish/lib/managers.hpp (revision ed3982131dcef2b499da36e674d2d21b2289ef29)
19c310685SBorawski.Lukasz /*
29c310685SBorawski.Lukasz // Copyright (c) 2018 Intel Corporation
39c310685SBorawski.Lukasz //
49c310685SBorawski.Lukasz // Licensed under the Apache License, Version 2.0 (the "License");
59c310685SBorawski.Lukasz // you may not use this file except in compliance with the License.
69c310685SBorawski.Lukasz // You may obtain a copy of the License at
79c310685SBorawski.Lukasz //
89c310685SBorawski.Lukasz //      http://www.apache.org/licenses/LICENSE-2.0
99c310685SBorawski.Lukasz //
109c310685SBorawski.Lukasz // Unless required by applicable law or agreed to in writing, software
119c310685SBorawski.Lukasz // distributed under the License is distributed on an "AS IS" BASIS,
129c310685SBorawski.Lukasz // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139c310685SBorawski.Lukasz // See the License for the specific language governing permissions and
149c310685SBorawski.Lukasz // limitations under the License.
159c310685SBorawski.Lukasz */
169c310685SBorawski.Lukasz #pragma once
179c310685SBorawski.Lukasz 
18b49ac873SJames Feist #include "health.hpp"
19c5d03ff4SJennifer Lee #include "redfish_util.hpp"
209c310685SBorawski.Lukasz 
217e860f15SJohn Edward Broadbent #include <app.hpp>
225b4aa86bSJames Feist #include <boost/algorithm/string/replace.hpp>
23af5d6058SSantosh Puranik #include <boost/date_time.hpp>
245b4aa86bSJames Feist #include <dbus_utility.hpp>
25*ed398213SEd Tanous #include <registries/privilege_registry.hpp>
26e90c5052SAndrew Geissler #include <utils/fw_utils.hpp>
277bffdb7eSBernard Wong #include <utils/systemd_utils.hpp>
281214b7e7SGunnar Mills 
294bfefa74SGunnar Mills #include <cstdint>
301214b7e7SGunnar Mills #include <memory>
311214b7e7SGunnar Mills #include <sstream>
32abf2add6SEd Tanous #include <variant>
335b4aa86bSJames Feist 
341abe55efSEd Tanous namespace redfish
351abe55efSEd Tanous {
36ed5befbdSJennifer Lee 
37ed5befbdSJennifer Lee /**
382a5c4407SGunnar Mills  * Function reboots the BMC.
392a5c4407SGunnar Mills  *
402a5c4407SGunnar Mills  * @param[in] asyncResp - Shared pointer for completing asynchronous calls
41ed5befbdSJennifer Lee  */
428d1b46d7Szhanghch05 inline void
438d1b46d7Szhanghch05     doBMCGracefulRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
44ed5befbdSJennifer Lee {
45ed5befbdSJennifer Lee     const char* processName = "xyz.openbmc_project.State.BMC";
46ed5befbdSJennifer Lee     const char* objectPath = "/xyz/openbmc_project/state/bmc0";
47ed5befbdSJennifer Lee     const char* interfaceName = "xyz.openbmc_project.State.BMC";
48ed5befbdSJennifer Lee     const std::string& propertyValue =
49ed5befbdSJennifer Lee         "xyz.openbmc_project.State.BMC.Transition.Reboot";
50ed5befbdSJennifer Lee     const char* destProperty = "RequestedBMCTransition";
51ed5befbdSJennifer Lee 
52ed5befbdSJennifer Lee     // Create the D-Bus variant for D-Bus call.
53ed5befbdSJennifer Lee     VariantType dbusPropertyValue(propertyValue);
54ed5befbdSJennifer Lee 
55ed5befbdSJennifer Lee     crow::connections::systemBus->async_method_call(
56ed5befbdSJennifer Lee         [asyncResp](const boost::system::error_code ec) {
57ed5befbdSJennifer Lee             // Use "Set" method to set the property value.
58ed5befbdSJennifer Lee             if (ec)
59ed5befbdSJennifer Lee             {
602a5c4407SGunnar Mills                 BMCWEB_LOG_DEBUG << "[Set] Bad D-Bus request error: " << ec;
61ed5befbdSJennifer Lee                 messages::internalError(asyncResp->res);
62ed5befbdSJennifer Lee                 return;
63ed5befbdSJennifer Lee             }
64ed5befbdSJennifer Lee 
65ed5befbdSJennifer Lee             messages::success(asyncResp->res);
66ed5befbdSJennifer Lee         },
67ed5befbdSJennifer Lee         processName, objectPath, "org.freedesktop.DBus.Properties", "Set",
68ed5befbdSJennifer Lee         interfaceName, destProperty, dbusPropertyValue);
69ed5befbdSJennifer Lee }
702a5c4407SGunnar Mills 
718d1b46d7Szhanghch05 inline void
728d1b46d7Szhanghch05     doBMCForceRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
73f92af389SJayaprakash Mutyala {
74f92af389SJayaprakash Mutyala     const char* processName = "xyz.openbmc_project.State.BMC";
75f92af389SJayaprakash Mutyala     const char* objectPath = "/xyz/openbmc_project/state/bmc0";
76f92af389SJayaprakash Mutyala     const char* interfaceName = "xyz.openbmc_project.State.BMC";
77f92af389SJayaprakash Mutyala     const std::string& propertyValue =
78f92af389SJayaprakash Mutyala         "xyz.openbmc_project.State.BMC.Transition.HardReboot";
79f92af389SJayaprakash Mutyala     const char* destProperty = "RequestedBMCTransition";
80f92af389SJayaprakash Mutyala 
81f92af389SJayaprakash Mutyala     // Create the D-Bus variant for D-Bus call.
82f92af389SJayaprakash Mutyala     VariantType dbusPropertyValue(propertyValue);
83f92af389SJayaprakash Mutyala 
84f92af389SJayaprakash Mutyala     crow::connections::systemBus->async_method_call(
85f92af389SJayaprakash Mutyala         [asyncResp](const boost::system::error_code ec) {
86f92af389SJayaprakash Mutyala             // Use "Set" method to set the property value.
87f92af389SJayaprakash Mutyala             if (ec)
88f92af389SJayaprakash Mutyala             {
89f92af389SJayaprakash Mutyala                 BMCWEB_LOG_DEBUG << "[Set] Bad D-Bus request error: " << ec;
90f92af389SJayaprakash Mutyala                 messages::internalError(asyncResp->res);
91f92af389SJayaprakash Mutyala                 return;
92f92af389SJayaprakash Mutyala             }
93f92af389SJayaprakash Mutyala 
94f92af389SJayaprakash Mutyala             messages::success(asyncResp->res);
95f92af389SJayaprakash Mutyala         },
96f92af389SJayaprakash Mutyala         processName, objectPath, "org.freedesktop.DBus.Properties", "Set",
97f92af389SJayaprakash Mutyala         interfaceName, destProperty, dbusPropertyValue);
98f92af389SJayaprakash Mutyala }
99f92af389SJayaprakash Mutyala 
1002a5c4407SGunnar Mills /**
1012a5c4407SGunnar Mills  * ManagerResetAction class supports the POST method for the Reset (reboot)
1022a5c4407SGunnar Mills  * action.
1032a5c4407SGunnar Mills  */
1047e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetAction(App& app)
1052a5c4407SGunnar Mills {
1062a5c4407SGunnar Mills     /**
1072a5c4407SGunnar Mills      * Function handles POST method request.
1082a5c4407SGunnar Mills      * Analyzes POST body before sending Reset (Reboot) request data to D-Bus.
109f92af389SJayaprakash Mutyala      * OpenBMC supports ResetType "GracefulRestart" and "ForceRestart".
1102a5c4407SGunnar Mills      */
1117e860f15SJohn Edward Broadbent 
1127e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/")
113*ed398213SEd Tanous         .privileges(redfish::privileges::postManager)
1147e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::post)(
1157e860f15SJohn Edward Broadbent             [](const crow::Request& req,
1167e860f15SJohn Edward Broadbent                const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
1172a5c4407SGunnar Mills                 BMCWEB_LOG_DEBUG << "Post Manager Reset.";
1182a5c4407SGunnar Mills 
1192a5c4407SGunnar Mills                 std::string resetType;
1202a5c4407SGunnar Mills 
1217e860f15SJohn Edward Broadbent                 if (!json_util::readJson(req, asyncResp->res, "ResetType",
1227e860f15SJohn Edward Broadbent                                          resetType))
1232a5c4407SGunnar Mills                 {
1242a5c4407SGunnar Mills                     return;
1252a5c4407SGunnar Mills                 }
1262a5c4407SGunnar Mills 
127f92af389SJayaprakash Mutyala                 if (resetType == "GracefulRestart")
128f92af389SJayaprakash Mutyala                 {
129f92af389SJayaprakash Mutyala                     BMCWEB_LOG_DEBUG << "Proceeding with " << resetType;
130f92af389SJayaprakash Mutyala                     doBMCGracefulRestart(asyncResp);
131f92af389SJayaprakash Mutyala                     return;
132f92af389SJayaprakash Mutyala                 }
1333174e4dfSEd Tanous                 if (resetType == "ForceRestart")
134f92af389SJayaprakash Mutyala                 {
135f92af389SJayaprakash Mutyala                     BMCWEB_LOG_DEBUG << "Proceeding with " << resetType;
136f92af389SJayaprakash Mutyala                     doBMCForceRestart(asyncResp);
137f92af389SJayaprakash Mutyala                     return;
138f92af389SJayaprakash Mutyala                 }
1392a5c4407SGunnar Mills                 BMCWEB_LOG_DEBUG << "Invalid property value for ResetType: "
1402a5c4407SGunnar Mills                                  << resetType;
1412a5c4407SGunnar Mills                 messages::actionParameterNotSupported(asyncResp->res, resetType,
1422a5c4407SGunnar Mills                                                       "ResetType");
1432a5c4407SGunnar Mills 
1442a5c4407SGunnar Mills                 return;
1457e860f15SJohn Edward Broadbent             });
1462a5c4407SGunnar Mills }
147ed5befbdSJennifer Lee 
1483e40fc74SGunnar Mills /**
1493e40fc74SGunnar Mills  * ManagerResetToDefaultsAction class supports POST method for factory reset
1503e40fc74SGunnar Mills  * action.
1513e40fc74SGunnar Mills  */
1527e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetToDefaultsAction(App& app)
1533e40fc74SGunnar Mills {
1543e40fc74SGunnar Mills 
1553e40fc74SGunnar Mills     /**
1563e40fc74SGunnar Mills      * Function handles ResetToDefaults POST method request.
1573e40fc74SGunnar Mills      *
1583e40fc74SGunnar Mills      * Analyzes POST body message and factory resets BMC by calling
1593e40fc74SGunnar Mills      * BMC code updater factory reset followed by a BMC reboot.
1603e40fc74SGunnar Mills      *
1613e40fc74SGunnar Mills      * BMC code updater factory reset wipes the whole BMC read-write
1623e40fc74SGunnar Mills      * filesystem which includes things like the network settings.
1633e40fc74SGunnar Mills      *
1643e40fc74SGunnar Mills      * OpenBMC only supports ResetToDefaultsType "ResetAll".
1653e40fc74SGunnar Mills      */
1667e860f15SJohn Edward Broadbent 
1677e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app,
1687e860f15SJohn Edward Broadbent                  "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults/")
169*ed398213SEd Tanous         .privileges(redfish::privileges::postManager)
1707e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::post)(
1717e860f15SJohn Edward Broadbent             [](const crow::Request& req,
1727e860f15SJohn Edward Broadbent                const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
1733e40fc74SGunnar Mills                 BMCWEB_LOG_DEBUG << "Post ResetToDefaults.";
1743e40fc74SGunnar Mills 
1753e40fc74SGunnar Mills                 std::string resetType;
1763e40fc74SGunnar Mills 
1777e860f15SJohn Edward Broadbent                 if (!json_util::readJson(req, asyncResp->res,
1787e860f15SJohn Edward Broadbent                                          "ResetToDefaultsType", resetType))
1793e40fc74SGunnar Mills                 {
1803e40fc74SGunnar Mills                     BMCWEB_LOG_DEBUG << "Missing property ResetToDefaultsType.";
1813e40fc74SGunnar Mills 
1827e860f15SJohn Edward Broadbent                     messages::actionParameterMissing(asyncResp->res,
1837e860f15SJohn Edward Broadbent                                                      "ResetToDefaults",
1843e40fc74SGunnar Mills                                                      "ResetToDefaultsType");
1853e40fc74SGunnar Mills                     return;
1863e40fc74SGunnar Mills                 }
1873e40fc74SGunnar Mills 
1883e40fc74SGunnar Mills                 if (resetType != "ResetAll")
1893e40fc74SGunnar Mills                 {
1903e40fc74SGunnar Mills                     BMCWEB_LOG_DEBUG << "Invalid property value for "
1913e40fc74SGunnar Mills                                         "ResetToDefaultsType: "
1923e40fc74SGunnar Mills                                      << resetType;
1937e860f15SJohn Edward Broadbent                     messages::actionParameterNotSupported(
1947e860f15SJohn Edward Broadbent                         asyncResp->res, resetType, "ResetToDefaultsType");
1953e40fc74SGunnar Mills                     return;
1963e40fc74SGunnar Mills                 }
1973e40fc74SGunnar Mills 
1983e40fc74SGunnar Mills                 crow::connections::systemBus->async_method_call(
1993e40fc74SGunnar Mills                     [asyncResp](const boost::system::error_code ec) {
2003e40fc74SGunnar Mills                         if (ec)
2013e40fc74SGunnar Mills                         {
2027e860f15SJohn Edward Broadbent                             BMCWEB_LOG_DEBUG << "Failed to ResetToDefaults: "
2037e860f15SJohn Edward Broadbent                                              << ec;
2043e40fc74SGunnar Mills                             messages::internalError(asyncResp->res);
2053e40fc74SGunnar Mills                             return;
2063e40fc74SGunnar Mills                         }
2073e40fc74SGunnar Mills                         // Factory Reset doesn't actually happen until a reboot
2083e40fc74SGunnar Mills                         // Can't erase what the BMC is running on
2093e40fc74SGunnar Mills                         doBMCGracefulRestart(asyncResp);
2103e40fc74SGunnar Mills                     },
2113e40fc74SGunnar Mills                     "xyz.openbmc_project.Software.BMC.Updater",
2123e40fc74SGunnar Mills                     "/xyz/openbmc_project/software",
2133e40fc74SGunnar Mills                     "xyz.openbmc_project.Common.FactoryReset", "Reset");
2147e860f15SJohn Edward Broadbent             });
2153e40fc74SGunnar Mills }
2163e40fc74SGunnar Mills 
2171cb1a9e6SAppaRao Puli /**
2181cb1a9e6SAppaRao Puli  * ManagerResetActionInfo derived class for delivering Manager
2191cb1a9e6SAppaRao Puli  * ResetType AllowableValues using ResetInfo schema.
2201cb1a9e6SAppaRao Puli  */
2217e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetActionInfo(App& app)
2221cb1a9e6SAppaRao Puli {
2231cb1a9e6SAppaRao Puli     /**
2241cb1a9e6SAppaRao Puli      * Functions triggers appropriate requests on DBus
2251cb1a9e6SAppaRao Puli      */
2267e860f15SJohn Edward Broadbent 
2277e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/ResetActionInfo/")
228*ed398213SEd Tanous         .privileges(redfish::privileges::getActionInfo)
2297e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::get)(
2307e860f15SJohn Edward Broadbent             [](const crow::Request&,
2317e860f15SJohn Edward Broadbent                const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2328d1b46d7Szhanghch05                 asyncResp->res.jsonValue = {
2331cb1a9e6SAppaRao Puli                     {"@odata.type", "#ActionInfo.v1_1_2.ActionInfo"},
2341cb1a9e6SAppaRao Puli                     {"@odata.id", "/redfish/v1/Managers/bmc/ResetActionInfo"},
2351cb1a9e6SAppaRao Puli                     {"Name", "Reset Action Info"},
2361cb1a9e6SAppaRao Puli                     {"Id", "ResetActionInfo"},
2371cb1a9e6SAppaRao Puli                     {"Parameters",
2381cb1a9e6SAppaRao Puli                      {{{"Name", "ResetType"},
2391cb1a9e6SAppaRao Puli                        {"Required", true},
2401cb1a9e6SAppaRao Puli                        {"DataType", "String"},
2417e860f15SJohn Edward Broadbent                        {"AllowableValues",
2427e860f15SJohn Edward Broadbent                         {"GracefulRestart", "ForceRestart"}}}}}};
2437e860f15SJohn Edward Broadbent             });
2441cb1a9e6SAppaRao Puli }
2451cb1a9e6SAppaRao Puli 
2465b4aa86bSJames Feist static constexpr const char* objectManagerIface =
2475b4aa86bSJames Feist     "org.freedesktop.DBus.ObjectManager";
2485b4aa86bSJames Feist static constexpr const char* pidConfigurationIface =
2495b4aa86bSJames Feist     "xyz.openbmc_project.Configuration.Pid";
2505b4aa86bSJames Feist static constexpr const char* pidZoneConfigurationIface =
2515b4aa86bSJames Feist     "xyz.openbmc_project.Configuration.Pid.Zone";
252b7a08d04SJames Feist static constexpr const char* stepwiseConfigurationIface =
253b7a08d04SJames Feist     "xyz.openbmc_project.Configuration.Stepwise";
25473df0db0SJames Feist static constexpr const char* thermalModeIface =
25573df0db0SJames Feist     "xyz.openbmc_project.Control.ThermalMode";
2569c310685SBorawski.Lukasz 
2578d1b46d7Szhanghch05 inline void
2588d1b46d7Szhanghch05     asyncPopulatePid(const std::string& connection, const std::string& path,
25973df0db0SJames Feist                      const std::string& currentProfile,
26073df0db0SJames Feist                      const std::vector<std::string>& supportedProfiles,
2618d1b46d7Szhanghch05                      const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
2625b4aa86bSJames Feist {
2635b4aa86bSJames Feist 
2645b4aa86bSJames Feist     crow::connections::systemBus->async_method_call(
26573df0db0SJames Feist         [asyncResp, currentProfile, supportedProfiles](
26673df0db0SJames Feist             const boost::system::error_code ec,
2675b4aa86bSJames Feist             const dbus::utility::ManagedObjectType& managedObj) {
2685b4aa86bSJames Feist             if (ec)
2695b4aa86bSJames Feist             {
2705b4aa86bSJames Feist                 BMCWEB_LOG_ERROR << ec;
2715b4aa86bSJames Feist                 asyncResp->res.jsonValue.clear();
272f12894f8SJason M. Bills                 messages::internalError(asyncResp->res);
2735b4aa86bSJames Feist                 return;
2745b4aa86bSJames Feist             }
2755b4aa86bSJames Feist             nlohmann::json& configRoot =
2765b4aa86bSJames Feist                 asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"];
2775b4aa86bSJames Feist             nlohmann::json& fans = configRoot["FanControllers"];
2785b4aa86bSJames Feist             fans["@odata.type"] = "#OemManager.FanControllers";
2795b4aa86bSJames Feist             fans["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/"
2805b4aa86bSJames Feist                                 "Fan/FanControllers";
2815b4aa86bSJames Feist 
2825b4aa86bSJames Feist             nlohmann::json& pids = configRoot["PidControllers"];
2835b4aa86bSJames Feist             pids["@odata.type"] = "#OemManager.PidControllers";
2845b4aa86bSJames Feist             pids["@odata.id"] =
2855b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers";
2865b4aa86bSJames Feist 
287b7a08d04SJames Feist             nlohmann::json& stepwise = configRoot["StepwiseControllers"];
288b7a08d04SJames Feist             stepwise["@odata.type"] = "#OemManager.StepwiseControllers";
289b7a08d04SJames Feist             stepwise["@odata.id"] =
290b7a08d04SJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers";
291b7a08d04SJames Feist 
2925b4aa86bSJames Feist             nlohmann::json& zones = configRoot["FanZones"];
2935b4aa86bSJames Feist             zones["@odata.id"] =
2945b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones";
2955b4aa86bSJames Feist             zones["@odata.type"] = "#OemManager.FanZones";
2965b4aa86bSJames Feist             configRoot["@odata.id"] =
2975b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan";
2985b4aa86bSJames Feist             configRoot["@odata.type"] = "#OemManager.Fan";
29973df0db0SJames Feist             configRoot["Profile@Redfish.AllowableValues"] = supportedProfiles;
30073df0db0SJames Feist 
30173df0db0SJames Feist             if (!currentProfile.empty())
30273df0db0SJames Feist             {
30373df0db0SJames Feist                 configRoot["Profile"] = currentProfile;
30473df0db0SJames Feist             }
30573df0db0SJames Feist             BMCWEB_LOG_ERROR << "profile = " << currentProfile << " !";
3065b4aa86bSJames Feist 
3075b4aa86bSJames Feist             for (const auto& pathPair : managedObj)
3085b4aa86bSJames Feist             {
3095b4aa86bSJames Feist                 for (const auto& intfPair : pathPair.second)
3105b4aa86bSJames Feist                 {
3115b4aa86bSJames Feist                     if (intfPair.first != pidConfigurationIface &&
312b7a08d04SJames Feist                         intfPair.first != pidZoneConfigurationIface &&
313b7a08d04SJames Feist                         intfPair.first != stepwiseConfigurationIface)
3145b4aa86bSJames Feist                     {
3155b4aa86bSJames Feist                         continue;
3165b4aa86bSJames Feist                     }
3175b4aa86bSJames Feist                     auto findName = intfPair.second.find("Name");
3185b4aa86bSJames Feist                     if (findName == intfPair.second.end())
3195b4aa86bSJames Feist                     {
3205b4aa86bSJames Feist                         BMCWEB_LOG_ERROR << "Pid Field missing Name";
321a08b46ccSJason M. Bills                         messages::internalError(asyncResp->res);
3225b4aa86bSJames Feist                         return;
3235b4aa86bSJames Feist                     }
32473df0db0SJames Feist 
3255b4aa86bSJames Feist                     const std::string* namePtr =
326abf2add6SEd Tanous                         std::get_if<std::string>(&findName->second);
3275b4aa86bSJames Feist                     if (namePtr == nullptr)
3285b4aa86bSJames Feist                     {
3295b4aa86bSJames Feist                         BMCWEB_LOG_ERROR << "Pid Name Field illegal";
330b7a08d04SJames Feist                         messages::internalError(asyncResp->res);
3315b4aa86bSJames Feist                         return;
3325b4aa86bSJames Feist                     }
3335b4aa86bSJames Feist                     std::string name = *namePtr;
3345b4aa86bSJames Feist                     dbus::utility::escapePathForDbus(name);
33573df0db0SJames Feist 
33673df0db0SJames Feist                     auto findProfiles = intfPair.second.find("Profiles");
33773df0db0SJames Feist                     if (findProfiles != intfPair.second.end())
33873df0db0SJames Feist                     {
33973df0db0SJames Feist                         const std::vector<std::string>* profiles =
34073df0db0SJames Feist                             std::get_if<std::vector<std::string>>(
34173df0db0SJames Feist                                 &findProfiles->second);
34273df0db0SJames Feist                         if (profiles == nullptr)
34373df0db0SJames Feist                         {
34473df0db0SJames Feist                             BMCWEB_LOG_ERROR << "Pid Profiles Field illegal";
34573df0db0SJames Feist                             messages::internalError(asyncResp->res);
34673df0db0SJames Feist                             return;
34773df0db0SJames Feist                         }
34873df0db0SJames Feist                         if (std::find(profiles->begin(), profiles->end(),
34973df0db0SJames Feist                                       currentProfile) == profiles->end())
35073df0db0SJames Feist                         {
35173df0db0SJames Feist                             BMCWEB_LOG_INFO
35273df0db0SJames Feist                                 << name << " not supported in current profile";
35373df0db0SJames Feist                             continue;
35473df0db0SJames Feist                         }
35573df0db0SJames Feist                     }
356b7a08d04SJames Feist                     nlohmann::json* config = nullptr;
357c33a90ecSJames Feist 
358c33a90ecSJames Feist                     const std::string* classPtr = nullptr;
359c33a90ecSJames Feist                     auto findClass = intfPair.second.find("Class");
360c33a90ecSJames Feist                     if (findClass != intfPair.second.end())
361c33a90ecSJames Feist                     {
362c33a90ecSJames Feist                         classPtr = std::get_if<std::string>(&findClass->second);
363c33a90ecSJames Feist                     }
364c33a90ecSJames Feist 
3655b4aa86bSJames Feist                     if (intfPair.first == pidZoneConfigurationIface)
3665b4aa86bSJames Feist                     {
3675b4aa86bSJames Feist                         std::string chassis;
3685b4aa86bSJames Feist                         if (!dbus::utility::getNthStringFromPath(
3695b4aa86bSJames Feist                                 pathPair.first.str, 5, chassis))
3705b4aa86bSJames Feist                         {
3715b4aa86bSJames Feist                             chassis = "#IllegalValue";
3725b4aa86bSJames Feist                         }
3735b4aa86bSJames Feist                         nlohmann::json& zone = zones[name];
3745b4aa86bSJames Feist                         zone["Chassis"] = {
3755b4aa86bSJames Feist                             {"@odata.id", "/redfish/v1/Chassis/" + chassis}};
3765b4aa86bSJames Feist                         zone["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/"
3775b4aa86bSJames Feist                                             "OpenBmc/Fan/FanZones/" +
3785b4aa86bSJames Feist                                             name;
3795b4aa86bSJames Feist                         zone["@odata.type"] = "#OemManager.FanZone";
380b7a08d04SJames Feist                         config = &zone;
3815b4aa86bSJames Feist                     }
3825b4aa86bSJames Feist 
383b7a08d04SJames Feist                     else if (intfPair.first == stepwiseConfigurationIface)
3845b4aa86bSJames Feist                     {
385c33a90ecSJames Feist                         if (classPtr == nullptr)
386c33a90ecSJames Feist                         {
387c33a90ecSJames Feist                             BMCWEB_LOG_ERROR << "Pid Class Field illegal";
388c33a90ecSJames Feist                             messages::internalError(asyncResp->res);
389c33a90ecSJames Feist                             return;
390c33a90ecSJames Feist                         }
391c33a90ecSJames Feist 
392b7a08d04SJames Feist                         nlohmann::json& controller = stepwise[name];
393b7a08d04SJames Feist                         config = &controller;
3945b4aa86bSJames Feist 
395b7a08d04SJames Feist                         controller["@odata.id"] =
396b7a08d04SJames Feist                             "/redfish/v1/Managers/bmc#/Oem/"
397b7a08d04SJames Feist                             "OpenBmc/Fan/StepwiseControllers/" +
398271584abSEd Tanous                             name;
399b7a08d04SJames Feist                         controller["@odata.type"] =
400b7a08d04SJames Feist                             "#OemManager.StepwiseController";
401b7a08d04SJames Feist 
402c33a90ecSJames Feist                         controller["Direction"] = *classPtr;
4035b4aa86bSJames Feist                     }
4045b4aa86bSJames Feist 
4055b4aa86bSJames Feist                     // pid and fans are off the same configuration
406b7a08d04SJames Feist                     else if (intfPair.first == pidConfigurationIface)
4075b4aa86bSJames Feist                     {
408c33a90ecSJames Feist 
4095b4aa86bSJames Feist                         if (classPtr == nullptr)
4105b4aa86bSJames Feist                         {
4115b4aa86bSJames Feist                             BMCWEB_LOG_ERROR << "Pid Class Field illegal";
412a08b46ccSJason M. Bills                             messages::internalError(asyncResp->res);
4135b4aa86bSJames Feist                             return;
4145b4aa86bSJames Feist                         }
4155b4aa86bSJames Feist                         bool isFan = *classPtr == "fan";
4165b4aa86bSJames Feist                         nlohmann::json& element =
4175b4aa86bSJames Feist                             isFan ? fans[name] : pids[name];
418b7a08d04SJames Feist                         config = &element;
4195b4aa86bSJames Feist                         if (isFan)
4205b4aa86bSJames Feist                         {
4215b4aa86bSJames Feist                             element["@odata.id"] =
4225b4aa86bSJames Feist                                 "/redfish/v1/Managers/bmc#/Oem/"
4235b4aa86bSJames Feist                                 "OpenBmc/Fan/FanControllers/" +
424271584abSEd Tanous                                 name;
4255b4aa86bSJames Feist                             element["@odata.type"] =
4265b4aa86bSJames Feist                                 "#OemManager.FanController";
4275b4aa86bSJames Feist                         }
4285b4aa86bSJames Feist                         else
4295b4aa86bSJames Feist                         {
4305b4aa86bSJames Feist                             element["@odata.id"] =
4315b4aa86bSJames Feist                                 "/redfish/v1/Managers/bmc#/Oem/"
4325b4aa86bSJames Feist                                 "OpenBmc/Fan/PidControllers/" +
433271584abSEd Tanous                                 name;
4345b4aa86bSJames Feist                             element["@odata.type"] =
4355b4aa86bSJames Feist                                 "#OemManager.PidController";
4365b4aa86bSJames Feist                         }
437b7a08d04SJames Feist                     }
438b7a08d04SJames Feist                     else
439b7a08d04SJames Feist                     {
440b7a08d04SJames Feist                         BMCWEB_LOG_ERROR << "Unexpected configuration";
441b7a08d04SJames Feist                         messages::internalError(asyncResp->res);
442b7a08d04SJames Feist                         return;
443b7a08d04SJames Feist                     }
444b7a08d04SJames Feist 
445b7a08d04SJames Feist                     // used for making maps out of 2 vectors
446b7a08d04SJames Feist                     const std::vector<double>* keys = nullptr;
447b7a08d04SJames Feist                     const std::vector<double>* values = nullptr;
448b7a08d04SJames Feist 
449b7a08d04SJames Feist                     for (const auto& propertyPair : intfPair.second)
450b7a08d04SJames Feist                     {
451b7a08d04SJames Feist                         if (propertyPair.first == "Type" ||
452b7a08d04SJames Feist                             propertyPair.first == "Class" ||
453b7a08d04SJames Feist                             propertyPair.first == "Name")
454b7a08d04SJames Feist                         {
455b7a08d04SJames Feist                             continue;
456b7a08d04SJames Feist                         }
457b7a08d04SJames Feist 
458b7a08d04SJames Feist                         // zones
459b7a08d04SJames Feist                         if (intfPair.first == pidZoneConfigurationIface)
460b7a08d04SJames Feist                         {
461b7a08d04SJames Feist                             const double* ptr =
462abf2add6SEd Tanous                                 std::get_if<double>(&propertyPair.second);
463b7a08d04SJames Feist                             if (ptr == nullptr)
464b7a08d04SJames Feist                             {
465b7a08d04SJames Feist                                 BMCWEB_LOG_ERROR << "Field Illegal "
466b7a08d04SJames Feist                                                  << propertyPair.first;
467b7a08d04SJames Feist                                 messages::internalError(asyncResp->res);
468b7a08d04SJames Feist                                 return;
469b7a08d04SJames Feist                             }
470b7a08d04SJames Feist                             (*config)[propertyPair.first] = *ptr;
471b7a08d04SJames Feist                         }
472b7a08d04SJames Feist 
473b7a08d04SJames Feist                         if (intfPair.first == stepwiseConfigurationIface)
474b7a08d04SJames Feist                         {
475b7a08d04SJames Feist                             if (propertyPair.first == "Reading" ||
476b7a08d04SJames Feist                                 propertyPair.first == "Output")
477b7a08d04SJames Feist                             {
478b7a08d04SJames Feist                                 const std::vector<double>* ptr =
479abf2add6SEd Tanous                                     std::get_if<std::vector<double>>(
480b7a08d04SJames Feist                                         &propertyPair.second);
481b7a08d04SJames Feist 
482b7a08d04SJames Feist                                 if (ptr == nullptr)
483b7a08d04SJames Feist                                 {
484b7a08d04SJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
485b7a08d04SJames Feist                                                      << propertyPair.first;
486b7a08d04SJames Feist                                     messages::internalError(asyncResp->res);
487b7a08d04SJames Feist                                     return;
488b7a08d04SJames Feist                                 }
489b7a08d04SJames Feist 
490b7a08d04SJames Feist                                 if (propertyPair.first == "Reading")
491b7a08d04SJames Feist                                 {
492b7a08d04SJames Feist                                     keys = ptr;
493b7a08d04SJames Feist                                 }
494b7a08d04SJames Feist                                 else
495b7a08d04SJames Feist                                 {
496b7a08d04SJames Feist                                     values = ptr;
497b7a08d04SJames Feist                                 }
498b7a08d04SJames Feist                                 if (keys && values)
499b7a08d04SJames Feist                                 {
500b7a08d04SJames Feist                                     if (keys->size() != values->size())
501b7a08d04SJames Feist                                     {
502b7a08d04SJames Feist                                         BMCWEB_LOG_ERROR
503b7a08d04SJames Feist                                             << "Reading and Output size don't "
504b7a08d04SJames Feist                                                "match ";
505b7a08d04SJames Feist                                         messages::internalError(asyncResp->res);
506b7a08d04SJames Feist                                         return;
507b7a08d04SJames Feist                                     }
508b7a08d04SJames Feist                                     nlohmann::json& steps = (*config)["Steps"];
509b7a08d04SJames Feist                                     steps = nlohmann::json::array();
510b7a08d04SJames Feist                                     for (size_t ii = 0; ii < keys->size(); ii++)
511b7a08d04SJames Feist                                     {
512b7a08d04SJames Feist                                         steps.push_back(
513b7a08d04SJames Feist                                             {{"Target", (*keys)[ii]},
514b7a08d04SJames Feist                                              {"Output", (*values)[ii]}});
515b7a08d04SJames Feist                                     }
516b7a08d04SJames Feist                                 }
517b7a08d04SJames Feist                             }
518b7a08d04SJames Feist                             if (propertyPair.first == "NegativeHysteresis" ||
519b7a08d04SJames Feist                                 propertyPair.first == "PositiveHysteresis")
520b7a08d04SJames Feist                             {
521b7a08d04SJames Feist                                 const double* ptr =
522abf2add6SEd Tanous                                     std::get_if<double>(&propertyPair.second);
523b7a08d04SJames Feist                                 if (ptr == nullptr)
524b7a08d04SJames Feist                                 {
525b7a08d04SJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
526b7a08d04SJames Feist                                                      << propertyPair.first;
527b7a08d04SJames Feist                                     messages::internalError(asyncResp->res);
528b7a08d04SJames Feist                                     return;
529b7a08d04SJames Feist                                 }
530b7a08d04SJames Feist                                 (*config)[propertyPair.first] = *ptr;
531b7a08d04SJames Feist                             }
532b7a08d04SJames Feist                         }
533b7a08d04SJames Feist 
534b7a08d04SJames Feist                         // pid and fans are off the same configuration
535b7a08d04SJames Feist                         if (intfPair.first == pidConfigurationIface ||
536b7a08d04SJames Feist                             intfPair.first == stepwiseConfigurationIface)
537b7a08d04SJames Feist                         {
5385b4aa86bSJames Feist 
5395b4aa86bSJames Feist                             if (propertyPair.first == "Zones")
5405b4aa86bSJames Feist                             {
5415b4aa86bSJames Feist                                 const std::vector<std::string>* inputs =
542abf2add6SEd Tanous                                     std::get_if<std::vector<std::string>>(
5431b6b96c5SEd Tanous                                         &propertyPair.second);
5445b4aa86bSJames Feist 
5455b4aa86bSJames Feist                                 if (inputs == nullptr)
5465b4aa86bSJames Feist                                 {
5475b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR
5485b4aa86bSJames Feist                                         << "Zones Pid Field Illegal";
549a08b46ccSJason M. Bills                                     messages::internalError(asyncResp->res);
5505b4aa86bSJames Feist                                     return;
5515b4aa86bSJames Feist                                 }
552b7a08d04SJames Feist                                 auto& data = (*config)[propertyPair.first];
5535b4aa86bSJames Feist                                 data = nlohmann::json::array();
5545b4aa86bSJames Feist                                 for (std::string itemCopy : *inputs)
5555b4aa86bSJames Feist                                 {
5565b4aa86bSJames Feist                                     dbus::utility::escapePathForDbus(itemCopy);
5575b4aa86bSJames Feist                                     data.push_back(
5585b4aa86bSJames Feist                                         {{"@odata.id",
5595b4aa86bSJames Feist                                           "/redfish/v1/Managers/bmc#/Oem/"
5605b4aa86bSJames Feist                                           "OpenBmc/Fan/FanZones/" +
5615b4aa86bSJames Feist                                               itemCopy}});
5625b4aa86bSJames Feist                                 }
5635b4aa86bSJames Feist                             }
5645b4aa86bSJames Feist                             // todo(james): may never happen, but this
5655b4aa86bSJames Feist                             // assumes configuration data referenced in the
5665b4aa86bSJames Feist                             // PID config is provided by the same daemon, we
5675b4aa86bSJames Feist                             // could add another loop to cover all cases,
5685b4aa86bSJames Feist                             // but I'm okay kicking this can down the road a
5695b4aa86bSJames Feist                             // bit
5705b4aa86bSJames Feist 
5715b4aa86bSJames Feist                             else if (propertyPair.first == "Inputs" ||
5725b4aa86bSJames Feist                                      propertyPair.first == "Outputs")
5735b4aa86bSJames Feist                             {
574b7a08d04SJames Feist                                 auto& data = (*config)[propertyPair.first];
5755b4aa86bSJames Feist                                 const std::vector<std::string>* inputs =
576abf2add6SEd Tanous                                     std::get_if<std::vector<std::string>>(
5771b6b96c5SEd Tanous                                         &propertyPair.second);
5785b4aa86bSJames Feist 
5795b4aa86bSJames Feist                                 if (inputs == nullptr)
5805b4aa86bSJames Feist                                 {
5815b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
5825b4aa86bSJames Feist                                                      << propertyPair.first;
583f12894f8SJason M. Bills                                     messages::internalError(asyncResp->res);
5845b4aa86bSJames Feist                                     return;
5855b4aa86bSJames Feist                                 }
5865b4aa86bSJames Feist                                 data = *inputs;
587b943aaefSJames Feist                             }
588b943aaefSJames Feist                             else if (propertyPair.first == "SetPointOffset")
589b943aaefSJames Feist                             {
590b943aaefSJames Feist                                 const std::string* ptr =
591b943aaefSJames Feist                                     std::get_if<std::string>(
592b943aaefSJames Feist                                         &propertyPair.second);
593b943aaefSJames Feist 
594b943aaefSJames Feist                                 if (ptr == nullptr)
595b943aaefSJames Feist                                 {
596b943aaefSJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
597b943aaefSJames Feist                                                      << propertyPair.first;
598b943aaefSJames Feist                                     messages::internalError(asyncResp->res);
599b943aaefSJames Feist                                     return;
600b943aaefSJames Feist                                 }
601b943aaefSJames Feist                                 // translate from dbus to redfish
602b943aaefSJames Feist                                 if (*ptr == "WarningHigh")
603b943aaefSJames Feist                                 {
604b943aaefSJames Feist                                     (*config)["SetPointOffset"] =
605b943aaefSJames Feist                                         "UpperThresholdNonCritical";
606b943aaefSJames Feist                                 }
607b943aaefSJames Feist                                 else if (*ptr == "WarningLow")
608b943aaefSJames Feist                                 {
609b943aaefSJames Feist                                     (*config)["SetPointOffset"] =
610b943aaefSJames Feist                                         "LowerThresholdNonCritical";
611b943aaefSJames Feist                                 }
612b943aaefSJames Feist                                 else if (*ptr == "CriticalHigh")
613b943aaefSJames Feist                                 {
614b943aaefSJames Feist                                     (*config)["SetPointOffset"] =
615b943aaefSJames Feist                                         "UpperThresholdCritical";
616b943aaefSJames Feist                                 }
617b943aaefSJames Feist                                 else if (*ptr == "CriticalLow")
618b943aaefSJames Feist                                 {
619b943aaefSJames Feist                                     (*config)["SetPointOffset"] =
620b943aaefSJames Feist                                         "LowerThresholdCritical";
621b943aaefSJames Feist                                 }
622b943aaefSJames Feist                                 else
623b943aaefSJames Feist                                 {
624b943aaefSJames Feist                                     BMCWEB_LOG_ERROR << "Value Illegal "
625b943aaefSJames Feist                                                      << *ptr;
626b943aaefSJames Feist                                     messages::internalError(asyncResp->res);
627b943aaefSJames Feist                                     return;
628b943aaefSJames Feist                                 }
629b943aaefSJames Feist                             }
630b943aaefSJames Feist                             // doubles
6315b4aa86bSJames Feist                             else if (propertyPair.first ==
6325b4aa86bSJames Feist                                          "FFGainCoefficient" ||
6335b4aa86bSJames Feist                                      propertyPair.first == "FFOffCoefficient" ||
6345b4aa86bSJames Feist                                      propertyPair.first == "ICoefficient" ||
6355b4aa86bSJames Feist                                      propertyPair.first == "ILimitMax" ||
6365b4aa86bSJames Feist                                      propertyPair.first == "ILimitMin" ||
637aad1a257SJames Feist                                      propertyPair.first ==
638aad1a257SJames Feist                                          "PositiveHysteresis" ||
639aad1a257SJames Feist                                      propertyPair.first ==
640aad1a257SJames Feist                                          "NegativeHysteresis" ||
6415b4aa86bSJames Feist                                      propertyPair.first == "OutLimitMax" ||
6425b4aa86bSJames Feist                                      propertyPair.first == "OutLimitMin" ||
6435b4aa86bSJames Feist                                      propertyPair.first == "PCoefficient" ||
6447625cb81SJames Feist                                      propertyPair.first == "SetPoint" ||
6455b4aa86bSJames Feist                                      propertyPair.first == "SlewNeg" ||
6465b4aa86bSJames Feist                                      propertyPair.first == "SlewPos")
6475b4aa86bSJames Feist                             {
6485b4aa86bSJames Feist                                 const double* ptr =
649abf2add6SEd Tanous                                     std::get_if<double>(&propertyPair.second);
6505b4aa86bSJames Feist                                 if (ptr == nullptr)
6515b4aa86bSJames Feist                                 {
6525b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
6535b4aa86bSJames Feist                                                      << propertyPair.first;
654f12894f8SJason M. Bills                                     messages::internalError(asyncResp->res);
6555b4aa86bSJames Feist                                     return;
6565b4aa86bSJames Feist                                 }
657b7a08d04SJames Feist                                 (*config)[propertyPair.first] = *ptr;
6585b4aa86bSJames Feist                             }
6595b4aa86bSJames Feist                         }
6605b4aa86bSJames Feist                     }
6615b4aa86bSJames Feist                 }
6625b4aa86bSJames Feist             }
6635b4aa86bSJames Feist         },
6645b4aa86bSJames Feist         connection, path, objectManagerIface, "GetManagedObjects");
6655b4aa86bSJames Feist }
666ca537928SJennifer Lee 
66783ff9ab6SJames Feist enum class CreatePIDRet
66883ff9ab6SJames Feist {
66983ff9ab6SJames Feist     fail,
67083ff9ab6SJames Feist     del,
67183ff9ab6SJames Feist     patch
67283ff9ab6SJames Feist };
67383ff9ab6SJames Feist 
6748d1b46d7Szhanghch05 inline bool
6758d1b46d7Szhanghch05     getZonesFromJsonReq(const std::shared_ptr<bmcweb::AsyncResp>& response,
6765f2caaefSJames Feist                         std::vector<nlohmann::json>& config,
6775f2caaefSJames Feist                         std::vector<std::string>& zones)
6785f2caaefSJames Feist {
679b6baeaa4SJames Feist     if (config.empty())
680b6baeaa4SJames Feist     {
681b6baeaa4SJames Feist         BMCWEB_LOG_ERROR << "Empty Zones";
682b6baeaa4SJames Feist         messages::propertyValueFormatError(response->res,
683b6baeaa4SJames Feist                                            nlohmann::json::array(), "Zones");
684b6baeaa4SJames Feist         return false;
685b6baeaa4SJames Feist     }
6865f2caaefSJames Feist     for (auto& odata : config)
6875f2caaefSJames Feist     {
6885f2caaefSJames Feist         std::string path;
6895f2caaefSJames Feist         if (!redfish::json_util::readJson(odata, response->res, "@odata.id",
6905f2caaefSJames Feist                                           path))
6915f2caaefSJames Feist         {
6925f2caaefSJames Feist             return false;
6935f2caaefSJames Feist         }
6945f2caaefSJames Feist         std::string input;
69561adbda3SJames Feist 
69661adbda3SJames Feist         // 8 below comes from
69761adbda3SJames Feist         // /redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/Left
69861adbda3SJames Feist         //     0    1     2      3    4    5      6     7      8
69961adbda3SJames Feist         if (!dbus::utility::getNthStringFromPath(path, 8, input))
7005f2caaefSJames Feist         {
7015f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Got invalid path " << path;
7025f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Illegal Type Zones";
7035f2caaefSJames Feist             messages::propertyValueFormatError(response->res, odata.dump(),
7045f2caaefSJames Feist                                                "Zones");
7055f2caaefSJames Feist             return false;
7065f2caaefSJames Feist         }
7075f2caaefSJames Feist         boost::replace_all(input, "_", " ");
7085f2caaefSJames Feist         zones.emplace_back(std::move(input));
7095f2caaefSJames Feist     }
7105f2caaefSJames Feist     return true;
7115f2caaefSJames Feist }
7125f2caaefSJames Feist 
71323a21a1cSEd Tanous inline const dbus::utility::ManagedItem*
71473df0db0SJames Feist     findChassis(const dbus::utility::ManagedObjectType& managedObj,
715b6baeaa4SJames Feist                 const std::string& value, std::string& chassis)
716b6baeaa4SJames Feist {
717b6baeaa4SJames Feist     BMCWEB_LOG_DEBUG << "Find Chassis: " << value << "\n";
718b6baeaa4SJames Feist 
719b6baeaa4SJames Feist     std::string escaped = boost::replace_all_copy(value, " ", "_");
720b6baeaa4SJames Feist     escaped = "/" + escaped;
721b6baeaa4SJames Feist     auto it = std::find_if(
722b6baeaa4SJames Feist         managedObj.begin(), managedObj.end(), [&escaped](const auto& obj) {
723b6baeaa4SJames Feist             if (boost::algorithm::ends_with(obj.first.str, escaped))
724b6baeaa4SJames Feist             {
725b6baeaa4SJames Feist                 BMCWEB_LOG_DEBUG << "Matched " << obj.first.str << "\n";
726b6baeaa4SJames Feist                 return true;
727b6baeaa4SJames Feist             }
728b6baeaa4SJames Feist             return false;
729b6baeaa4SJames Feist         });
730b6baeaa4SJames Feist 
731b6baeaa4SJames Feist     if (it == managedObj.end())
732b6baeaa4SJames Feist     {
73373df0db0SJames Feist         return nullptr;
734b6baeaa4SJames Feist     }
735b6baeaa4SJames Feist     // 5 comes from <chassis-name> being the 5th element
736b6baeaa4SJames Feist     // /xyz/openbmc_project/inventory/system/chassis/<chassis-name>
73773df0db0SJames Feist     if (dbus::utility::getNthStringFromPath(it->first.str, 5, chassis))
73873df0db0SJames Feist     {
73973df0db0SJames Feist         return &(*it);
74073df0db0SJames Feist     }
74173df0db0SJames Feist 
74273df0db0SJames Feist     return nullptr;
743b6baeaa4SJames Feist }
744b6baeaa4SJames Feist 
74523a21a1cSEd Tanous inline CreatePIDRet createPidInterface(
7468d1b46d7Szhanghch05     const std::shared_ptr<bmcweb::AsyncResp>& response, const std::string& type,
747b5a76932SEd Tanous     const nlohmann::json::iterator& it, const std::string& path,
74883ff9ab6SJames Feist     const dbus::utility::ManagedObjectType& managedObj, bool createNewObject,
74983ff9ab6SJames Feist     boost::container::flat_map<std::string, dbus::utility::DbusVariantType>&
75083ff9ab6SJames Feist         output,
75173df0db0SJames Feist     std::string& chassis, const std::string& profile)
75283ff9ab6SJames Feist {
75383ff9ab6SJames Feist 
7545f2caaefSJames Feist     // common deleter
755b6baeaa4SJames Feist     if (it.value() == nullptr)
7565f2caaefSJames Feist     {
7575f2caaefSJames Feist         std::string iface;
7585f2caaefSJames Feist         if (type == "PidControllers" || type == "FanControllers")
7595f2caaefSJames Feist         {
7605f2caaefSJames Feist             iface = pidConfigurationIface;
7615f2caaefSJames Feist         }
7625f2caaefSJames Feist         else if (type == "FanZones")
7635f2caaefSJames Feist         {
7645f2caaefSJames Feist             iface = pidZoneConfigurationIface;
7655f2caaefSJames Feist         }
7665f2caaefSJames Feist         else if (type == "StepwiseControllers")
7675f2caaefSJames Feist         {
7685f2caaefSJames Feist             iface = stepwiseConfigurationIface;
7695f2caaefSJames Feist         }
7705f2caaefSJames Feist         else
7715f2caaefSJames Feist         {
772a0744d38SGunnar Mills             BMCWEB_LOG_ERROR << "Illegal Type " << type;
7735f2caaefSJames Feist             messages::propertyUnknown(response->res, type);
7745f2caaefSJames Feist             return CreatePIDRet::fail;
7755f2caaefSJames Feist         }
7766ee7f774SJames Feist 
7776ee7f774SJames Feist         BMCWEB_LOG_DEBUG << "del " << path << " " << iface << "\n";
7785f2caaefSJames Feist         // delete interface
7795f2caaefSJames Feist         crow::connections::systemBus->async_method_call(
7805f2caaefSJames Feist             [response, path](const boost::system::error_code ec) {
7815f2caaefSJames Feist                 if (ec)
7825f2caaefSJames Feist                 {
7835f2caaefSJames Feist                     BMCWEB_LOG_ERROR << "Error patching " << path << ": " << ec;
7845f2caaefSJames Feist                     messages::internalError(response->res);
785b6baeaa4SJames Feist                     return;
7865f2caaefSJames Feist                 }
787b6baeaa4SJames Feist                 messages::success(response->res);
7885f2caaefSJames Feist             },
7895f2caaefSJames Feist             "xyz.openbmc_project.EntityManager", path, iface, "Delete");
7905f2caaefSJames Feist         return CreatePIDRet::del;
7915f2caaefSJames Feist     }
7925f2caaefSJames Feist 
79373df0db0SJames Feist     const dbus::utility::ManagedItem* managedItem = nullptr;
794b6baeaa4SJames Feist     if (!createNewObject)
795b6baeaa4SJames Feist     {
796b6baeaa4SJames Feist         // if we aren't creating a new object, we should be able to find it on
797b6baeaa4SJames Feist         // d-bus
79873df0db0SJames Feist         managedItem = findChassis(managedObj, it.key(), chassis);
79973df0db0SJames Feist         if (managedItem == nullptr)
800b6baeaa4SJames Feist         {
801b6baeaa4SJames Feist             BMCWEB_LOG_ERROR << "Failed to get chassis from config patch";
802b6baeaa4SJames Feist             messages::invalidObject(response->res, it.key());
803b6baeaa4SJames Feist             return CreatePIDRet::fail;
804b6baeaa4SJames Feist         }
805b6baeaa4SJames Feist     }
806b6baeaa4SJames Feist 
80773df0db0SJames Feist     if (profile.size() &&
80873df0db0SJames Feist         (type == "PidControllers" || type == "FanControllers" ||
80973df0db0SJames Feist          type == "StepwiseControllers"))
81073df0db0SJames Feist     {
81173df0db0SJames Feist         if (managedItem == nullptr)
81273df0db0SJames Feist         {
81373df0db0SJames Feist             output["Profiles"] = std::vector<std::string>{profile};
81473df0db0SJames Feist         }
81573df0db0SJames Feist         else
81673df0db0SJames Feist         {
81773df0db0SJames Feist             std::string interface;
81873df0db0SJames Feist             if (type == "StepwiseControllers")
81973df0db0SJames Feist             {
82073df0db0SJames Feist                 interface = stepwiseConfigurationIface;
82173df0db0SJames Feist             }
82273df0db0SJames Feist             else
82373df0db0SJames Feist             {
82473df0db0SJames Feist                 interface = pidConfigurationIface;
82573df0db0SJames Feist             }
82673df0db0SJames Feist             auto findConfig = managedItem->second.find(interface);
82773df0db0SJames Feist             if (findConfig == managedItem->second.end())
82873df0db0SJames Feist             {
82973df0db0SJames Feist                 BMCWEB_LOG_ERROR
83073df0db0SJames Feist                     << "Failed to find interface in managed object";
83173df0db0SJames Feist                 messages::internalError(response->res);
83273df0db0SJames Feist                 return CreatePIDRet::fail;
83373df0db0SJames Feist             }
83473df0db0SJames Feist             auto findProfiles = findConfig->second.find("Profiles");
83573df0db0SJames Feist             if (findProfiles != findConfig->second.end())
83673df0db0SJames Feist             {
83773df0db0SJames Feist                 const std::vector<std::string>* curProfiles =
83873df0db0SJames Feist                     std::get_if<std::vector<std::string>>(
83973df0db0SJames Feist                         &(findProfiles->second));
84073df0db0SJames Feist                 if (curProfiles == nullptr)
84173df0db0SJames Feist                 {
84273df0db0SJames Feist                     BMCWEB_LOG_ERROR << "Illegal profiles in managed object";
84373df0db0SJames Feist                     messages::internalError(response->res);
84473df0db0SJames Feist                     return CreatePIDRet::fail;
84573df0db0SJames Feist                 }
84673df0db0SJames Feist                 if (std::find(curProfiles->begin(), curProfiles->end(),
84773df0db0SJames Feist                               profile) == curProfiles->end())
84873df0db0SJames Feist                 {
84973df0db0SJames Feist                     std::vector<std::string> newProfiles = *curProfiles;
85073df0db0SJames Feist                     newProfiles.push_back(profile);
85173df0db0SJames Feist                     output["Profiles"] = newProfiles;
85273df0db0SJames Feist                 }
85373df0db0SJames Feist             }
85473df0db0SJames Feist         }
85573df0db0SJames Feist     }
85673df0db0SJames Feist 
85783ff9ab6SJames Feist     if (type == "PidControllers" || type == "FanControllers")
85883ff9ab6SJames Feist     {
85983ff9ab6SJames Feist         if (createNewObject)
86083ff9ab6SJames Feist         {
86183ff9ab6SJames Feist             output["Class"] = type == "PidControllers" ? std::string("temp")
86283ff9ab6SJames Feist                                                        : std::string("fan");
86383ff9ab6SJames Feist             output["Type"] = std::string("Pid");
86483ff9ab6SJames Feist         }
8655f2caaefSJames Feist 
8665f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> zones;
8675f2caaefSJames Feist         std::optional<std::vector<std::string>> inputs;
8685f2caaefSJames Feist         std::optional<std::vector<std::string>> outputs;
8695f2caaefSJames Feist         std::map<std::string, std::optional<double>> doubles;
870b943aaefSJames Feist         std::optional<std::string> setpointOffset;
8715f2caaefSJames Feist         if (!redfish::json_util::readJson(
872b6baeaa4SJames Feist                 it.value(), response->res, "Inputs", inputs, "Outputs", outputs,
8735f2caaefSJames Feist                 "Zones", zones, "FFGainCoefficient",
8745f2caaefSJames Feist                 doubles["FFGainCoefficient"], "FFOffCoefficient",
8755f2caaefSJames Feist                 doubles["FFOffCoefficient"], "ICoefficient",
8765f2caaefSJames Feist                 doubles["ICoefficient"], "ILimitMax", doubles["ILimitMax"],
8775f2caaefSJames Feist                 "ILimitMin", doubles["ILimitMin"], "OutLimitMax",
8785f2caaefSJames Feist                 doubles["OutLimitMax"], "OutLimitMin", doubles["OutLimitMin"],
8795f2caaefSJames Feist                 "PCoefficient", doubles["PCoefficient"], "SetPoint",
880b943aaefSJames Feist                 doubles["SetPoint"], "SetPointOffset", setpointOffset,
881b943aaefSJames Feist                 "SlewNeg", doubles["SlewNeg"], "SlewPos", doubles["SlewPos"],
882b943aaefSJames Feist                 "PositiveHysteresis", doubles["PositiveHysteresis"],
883b943aaefSJames Feist                 "NegativeHysteresis", doubles["NegativeHysteresis"]))
88483ff9ab6SJames Feist         {
88571f52d96SEd Tanous             BMCWEB_LOG_ERROR
88671f52d96SEd Tanous                 << "Illegal Property "
88771f52d96SEd Tanous                 << it.value().dump(2, ' ', true,
88871f52d96SEd Tanous                                    nlohmann::json::error_handler_t::replace);
8895f2caaefSJames Feist             return CreatePIDRet::fail;
89083ff9ab6SJames Feist         }
8915f2caaefSJames Feist         if (zones)
8925f2caaefSJames Feist         {
8935f2caaefSJames Feist             std::vector<std::string> zonesStr;
8945f2caaefSJames Feist             if (!getZonesFromJsonReq(response, *zones, zonesStr))
8955f2caaefSJames Feist             {
896a0744d38SGunnar Mills                 BMCWEB_LOG_ERROR << "Illegal Zones";
8975f2caaefSJames Feist                 return CreatePIDRet::fail;
8985f2caaefSJames Feist             }
899b6baeaa4SJames Feist             if (chassis.empty() &&
900b6baeaa4SJames Feist                 !findChassis(managedObj, zonesStr[0], chassis))
901b6baeaa4SJames Feist             {
902b6baeaa4SJames Feist                 BMCWEB_LOG_ERROR << "Failed to get chassis from config patch";
903b6baeaa4SJames Feist                 messages::invalidObject(response->res, it.key());
904b6baeaa4SJames Feist                 return CreatePIDRet::fail;
905b6baeaa4SJames Feist             }
906b6baeaa4SJames Feist 
9075f2caaefSJames Feist             output["Zones"] = std::move(zonesStr);
9085f2caaefSJames Feist         }
9095f2caaefSJames Feist         if (inputs || outputs)
9105f2caaefSJames Feist         {
9115f2caaefSJames Feist             std::array<std::optional<std::vector<std::string>>*, 2> containers =
9125f2caaefSJames Feist                 {&inputs, &outputs};
9135f2caaefSJames Feist             size_t index = 0;
9145f2caaefSJames Feist             for (const auto& containerPtr : containers)
9155f2caaefSJames Feist             {
9165f2caaefSJames Feist                 std::optional<std::vector<std::string>>& container =
9175f2caaefSJames Feist                     *containerPtr;
9185f2caaefSJames Feist                 if (!container)
9195f2caaefSJames Feist                 {
9205f2caaefSJames Feist                     index++;
9215f2caaefSJames Feist                     continue;
92283ff9ab6SJames Feist                 }
92383ff9ab6SJames Feist 
9245f2caaefSJames Feist                 for (std::string& value : *container)
92583ff9ab6SJames Feist                 {
9265f2caaefSJames Feist                     boost::replace_all(value, "_", " ");
92783ff9ab6SJames Feist                 }
9285f2caaefSJames Feist                 std::string key;
9295f2caaefSJames Feist                 if (index == 0)
9305f2caaefSJames Feist                 {
9315f2caaefSJames Feist                     key = "Inputs";
9325f2caaefSJames Feist                 }
9335f2caaefSJames Feist                 else
9345f2caaefSJames Feist                 {
9355f2caaefSJames Feist                     key = "Outputs";
9365f2caaefSJames Feist                 }
9375f2caaefSJames Feist                 output[key] = *container;
9385f2caaefSJames Feist                 index++;
9395f2caaefSJames Feist             }
94083ff9ab6SJames Feist         }
94183ff9ab6SJames Feist 
942b943aaefSJames Feist         if (setpointOffset)
943b943aaefSJames Feist         {
944b943aaefSJames Feist             // translate between redfish and dbus names
945b943aaefSJames Feist             if (*setpointOffset == "UpperThresholdNonCritical")
946b943aaefSJames Feist             {
947b943aaefSJames Feist                 output["SetPointOffset"] = std::string("WarningLow");
948b943aaefSJames Feist             }
949b943aaefSJames Feist             else if (*setpointOffset == "LowerThresholdNonCritical")
950b943aaefSJames Feist             {
951b943aaefSJames Feist                 output["SetPointOffset"] = std::string("WarningHigh");
952b943aaefSJames Feist             }
953b943aaefSJames Feist             else if (*setpointOffset == "LowerThresholdCritical")
954b943aaefSJames Feist             {
955b943aaefSJames Feist                 output["SetPointOffset"] = std::string("CriticalLow");
956b943aaefSJames Feist             }
957b943aaefSJames Feist             else if (*setpointOffset == "UpperThresholdCritical")
958b943aaefSJames Feist             {
959b943aaefSJames Feist                 output["SetPointOffset"] = std::string("CriticalHigh");
960b943aaefSJames Feist             }
961b943aaefSJames Feist             else
962b943aaefSJames Feist             {
963b943aaefSJames Feist                 BMCWEB_LOG_ERROR << "Invalid setpointoffset "
964b943aaefSJames Feist                                  << *setpointOffset;
965b943aaefSJames Feist                 messages::invalidObject(response->res, it.key());
966b943aaefSJames Feist                 return CreatePIDRet::fail;
967b943aaefSJames Feist             }
968b943aaefSJames Feist         }
969b943aaefSJames Feist 
97083ff9ab6SJames Feist         // doubles
9715f2caaefSJames Feist         for (const auto& pairs : doubles)
97283ff9ab6SJames Feist         {
9735f2caaefSJames Feist             if (!pairs.second)
97483ff9ab6SJames Feist             {
9755f2caaefSJames Feist                 continue;
97683ff9ab6SJames Feist             }
9775f2caaefSJames Feist             BMCWEB_LOG_DEBUG << pairs.first << " = " << *pairs.second;
9785f2caaefSJames Feist             output[pairs.first] = *(pairs.second);
9795f2caaefSJames Feist         }
98083ff9ab6SJames Feist     }
98183ff9ab6SJames Feist 
98283ff9ab6SJames Feist     else if (type == "FanZones")
98383ff9ab6SJames Feist     {
98483ff9ab6SJames Feist         output["Type"] = std::string("Pid.Zone");
98583ff9ab6SJames Feist 
9865f2caaefSJames Feist         std::optional<nlohmann::json> chassisContainer;
9875f2caaefSJames Feist         std::optional<double> failSafePercent;
988d3ec07f8SJames Feist         std::optional<double> minThermalOutput;
989b6baeaa4SJames Feist         if (!redfish::json_util::readJson(it.value(), response->res, "Chassis",
9905f2caaefSJames Feist                                           chassisContainer, "FailSafePercent",
991d3ec07f8SJames Feist                                           failSafePercent, "MinThermalOutput",
992d3ec07f8SJames Feist                                           minThermalOutput))
99383ff9ab6SJames Feist         {
99471f52d96SEd Tanous             BMCWEB_LOG_ERROR
99571f52d96SEd Tanous                 << "Illegal Property "
99671f52d96SEd Tanous                 << it.value().dump(2, ' ', true,
99771f52d96SEd Tanous                                    nlohmann::json::error_handler_t::replace);
99883ff9ab6SJames Feist             return CreatePIDRet::fail;
99983ff9ab6SJames Feist         }
10005f2caaefSJames Feist 
10015f2caaefSJames Feist         if (chassisContainer)
100283ff9ab6SJames Feist         {
10035f2caaefSJames Feist 
10045f2caaefSJames Feist             std::string chassisId;
10055f2caaefSJames Feist             if (!redfish::json_util::readJson(*chassisContainer, response->res,
10065f2caaefSJames Feist                                               "@odata.id", chassisId))
10075f2caaefSJames Feist             {
100871f52d96SEd Tanous                 BMCWEB_LOG_ERROR
100971f52d96SEd Tanous                     << "Illegal Property "
101071f52d96SEd Tanous                     << chassisContainer->dump(
101171f52d96SEd Tanous                            2, ' ', true,
101271f52d96SEd Tanous                            nlohmann::json::error_handler_t::replace);
101383ff9ab6SJames Feist                 return CreatePIDRet::fail;
101483ff9ab6SJames Feist             }
101583ff9ab6SJames Feist 
1016717794d5SAppaRao Puli             // /redfish/v1/chassis/chassis_name/
10175f2caaefSJames Feist             if (!dbus::utility::getNthStringFromPath(chassisId, 3, chassis))
101883ff9ab6SJames Feist             {
10195f2caaefSJames Feist                 BMCWEB_LOG_ERROR << "Got invalid path " << chassisId;
10205f2caaefSJames Feist                 messages::invalidObject(response->res, chassisId);
102183ff9ab6SJames Feist                 return CreatePIDRet::fail;
102283ff9ab6SJames Feist             }
102383ff9ab6SJames Feist         }
1024d3ec07f8SJames Feist         if (minThermalOutput)
102583ff9ab6SJames Feist         {
1026d3ec07f8SJames Feist             output["MinThermalOutput"] = *minThermalOutput;
10275f2caaefSJames Feist         }
10285f2caaefSJames Feist         if (failSafePercent)
102983ff9ab6SJames Feist         {
10305f2caaefSJames Feist             output["FailSafePercent"] = *failSafePercent;
10315f2caaefSJames Feist         }
10325f2caaefSJames Feist     }
10335f2caaefSJames Feist     else if (type == "StepwiseControllers")
10345f2caaefSJames Feist     {
10355f2caaefSJames Feist         output["Type"] = std::string("Stepwise");
10365f2caaefSJames Feist 
10375f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> zones;
10385f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> steps;
10395f2caaefSJames Feist         std::optional<std::vector<std::string>> inputs;
10405f2caaefSJames Feist         std::optional<double> positiveHysteresis;
10415f2caaefSJames Feist         std::optional<double> negativeHysteresis;
1042c33a90ecSJames Feist         std::optional<std::string> direction; // upper clipping curve vs lower
10435f2caaefSJames Feist         if (!redfish::json_util::readJson(
1044b6baeaa4SJames Feist                 it.value(), response->res, "Zones", zones, "Steps", steps,
1045b6baeaa4SJames Feist                 "Inputs", inputs, "PositiveHysteresis", positiveHysteresis,
1046c33a90ecSJames Feist                 "NegativeHysteresis", negativeHysteresis, "Direction",
1047c33a90ecSJames Feist                 direction))
10485f2caaefSJames Feist         {
104971f52d96SEd Tanous             BMCWEB_LOG_ERROR
105071f52d96SEd Tanous                 << "Illegal Property "
105171f52d96SEd Tanous                 << it.value().dump(2, ' ', true,
105271f52d96SEd Tanous                                    nlohmann::json::error_handler_t::replace);
105383ff9ab6SJames Feist             return CreatePIDRet::fail;
105483ff9ab6SJames Feist         }
10555f2caaefSJames Feist 
10565f2caaefSJames Feist         if (zones)
105783ff9ab6SJames Feist         {
1058b6baeaa4SJames Feist             std::vector<std::string> zonesStrs;
1059b6baeaa4SJames Feist             if (!getZonesFromJsonReq(response, *zones, zonesStrs))
10605f2caaefSJames Feist             {
1061a0744d38SGunnar Mills                 BMCWEB_LOG_ERROR << "Illegal Zones";
106283ff9ab6SJames Feist                 return CreatePIDRet::fail;
106383ff9ab6SJames Feist             }
1064b6baeaa4SJames Feist             if (chassis.empty() &&
1065b6baeaa4SJames Feist                 !findChassis(managedObj, zonesStrs[0], chassis))
1066b6baeaa4SJames Feist             {
1067b6baeaa4SJames Feist                 BMCWEB_LOG_ERROR << "Failed to get chassis from config patch";
1068b6baeaa4SJames Feist                 messages::invalidObject(response->res, it.key());
1069b6baeaa4SJames Feist                 return CreatePIDRet::fail;
1070b6baeaa4SJames Feist             }
1071b6baeaa4SJames Feist             output["Zones"] = std::move(zonesStrs);
10725f2caaefSJames Feist         }
10735f2caaefSJames Feist         if (steps)
10745f2caaefSJames Feist         {
10755f2caaefSJames Feist             std::vector<double> readings;
10765f2caaefSJames Feist             std::vector<double> outputs;
10775f2caaefSJames Feist             for (auto& step : *steps)
10785f2caaefSJames Feist             {
10795f2caaefSJames Feist                 double target;
108023a21a1cSEd Tanous                 double out;
10815f2caaefSJames Feist 
10825f2caaefSJames Feist                 if (!redfish::json_util::readJson(step, response->res, "Target",
108323a21a1cSEd Tanous                                                   target, "Output", out))
10845f2caaefSJames Feist                 {
108571f52d96SEd Tanous                     BMCWEB_LOG_ERROR
108671f52d96SEd Tanous                         << "Illegal Property "
108771f52d96SEd Tanous                         << it.value().dump(
108871f52d96SEd Tanous                                2, ' ', true,
108971f52d96SEd Tanous                                nlohmann::json::error_handler_t::replace);
10905f2caaefSJames Feist                     return CreatePIDRet::fail;
10915f2caaefSJames Feist                 }
10925f2caaefSJames Feist                 readings.emplace_back(target);
109323a21a1cSEd Tanous                 outputs.emplace_back(out);
10945f2caaefSJames Feist             }
10955f2caaefSJames Feist             output["Reading"] = std::move(readings);
10965f2caaefSJames Feist             output["Output"] = std::move(outputs);
10975f2caaefSJames Feist         }
10985f2caaefSJames Feist         if (inputs)
10995f2caaefSJames Feist         {
11005f2caaefSJames Feist             for (std::string& value : *inputs)
11015f2caaefSJames Feist             {
11025f2caaefSJames Feist                 boost::replace_all(value, "_", " ");
11035f2caaefSJames Feist             }
11045f2caaefSJames Feist             output["Inputs"] = std::move(*inputs);
11055f2caaefSJames Feist         }
11065f2caaefSJames Feist         if (negativeHysteresis)
11075f2caaefSJames Feist         {
11085f2caaefSJames Feist             output["NegativeHysteresis"] = *negativeHysteresis;
11095f2caaefSJames Feist         }
11105f2caaefSJames Feist         if (positiveHysteresis)
11115f2caaefSJames Feist         {
11125f2caaefSJames Feist             output["PositiveHysteresis"] = *positiveHysteresis;
111383ff9ab6SJames Feist         }
1114c33a90ecSJames Feist         if (direction)
1115c33a90ecSJames Feist         {
1116c33a90ecSJames Feist             constexpr const std::array<const char*, 2> allowedDirections = {
1117c33a90ecSJames Feist                 "Ceiling", "Floor"};
1118c33a90ecSJames Feist             if (std::find(allowedDirections.begin(), allowedDirections.end(),
1119c33a90ecSJames Feist                           *direction) == allowedDirections.end())
1120c33a90ecSJames Feist             {
1121c33a90ecSJames Feist                 messages::propertyValueTypeError(response->res, "Direction",
1122c33a90ecSJames Feist                                                  *direction);
1123c33a90ecSJames Feist                 return CreatePIDRet::fail;
1124c33a90ecSJames Feist             }
1125c33a90ecSJames Feist             output["Class"] = *direction;
1126c33a90ecSJames Feist         }
112783ff9ab6SJames Feist     }
112883ff9ab6SJames Feist     else
112983ff9ab6SJames Feist     {
1130a0744d38SGunnar Mills         BMCWEB_LOG_ERROR << "Illegal Type " << type;
113135a62c7cSJason M. Bills         messages::propertyUnknown(response->res, type);
113283ff9ab6SJames Feist         return CreatePIDRet::fail;
113383ff9ab6SJames Feist     }
113483ff9ab6SJames Feist     return CreatePIDRet::patch;
113583ff9ab6SJames Feist }
113673df0db0SJames Feist struct GetPIDValues : std::enable_shared_from_this<GetPIDValues>
113773df0db0SJames Feist {
113883ff9ab6SJames Feist 
11398d1b46d7Szhanghch05     GetPIDValues(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn) :
114023a21a1cSEd Tanous         asyncResp(asyncRespIn)
114173df0db0SJames Feist 
11421214b7e7SGunnar Mills     {}
11439c310685SBorawski.Lukasz 
114473df0db0SJames Feist     void run()
11455b4aa86bSJames Feist     {
114673df0db0SJames Feist         std::shared_ptr<GetPIDValues> self = shared_from_this();
114773df0db0SJames Feist 
114873df0db0SJames Feist         // get all configurations
11495b4aa86bSJames Feist         crow::connections::systemBus->async_method_call(
115073df0db0SJames Feist             [self](const boost::system::error_code ec,
115123a21a1cSEd Tanous                    const crow::openbmc_mapper::GetSubTreeType& subtreeLocal) {
11525b4aa86bSJames Feist                 if (ec)
11535b4aa86bSJames Feist                 {
11545b4aa86bSJames Feist                     BMCWEB_LOG_ERROR << ec;
115573df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
115673df0db0SJames Feist                     return;
115773df0db0SJames Feist                 }
115823a21a1cSEd Tanous                 self->subtree = subtreeLocal;
115973df0db0SJames Feist             },
116073df0db0SJames Feist             "xyz.openbmc_project.ObjectMapper",
116173df0db0SJames Feist             "/xyz/openbmc_project/object_mapper",
116273df0db0SJames Feist             "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0,
116373df0db0SJames Feist             std::array<const char*, 4>{
116473df0db0SJames Feist                 pidConfigurationIface, pidZoneConfigurationIface,
116573df0db0SJames Feist                 objectManagerIface, stepwiseConfigurationIface});
116673df0db0SJames Feist 
116773df0db0SJames Feist         // at the same time get the selected profile
116873df0db0SJames Feist         crow::connections::systemBus->async_method_call(
116973df0db0SJames Feist             [self](const boost::system::error_code ec,
117023a21a1cSEd Tanous                    const crow::openbmc_mapper::GetSubTreeType& subtreeLocal) {
117123a21a1cSEd Tanous                 if (ec || subtreeLocal.empty())
117273df0db0SJames Feist                 {
117373df0db0SJames Feist                     return;
117473df0db0SJames Feist                 }
117523a21a1cSEd Tanous                 if (subtreeLocal[0].second.size() != 1)
117673df0db0SJames Feist                 {
117773df0db0SJames Feist                     // invalid mapper response, should never happen
117873df0db0SJames Feist                     BMCWEB_LOG_ERROR << "GetPIDValues: Mapper Error";
117973df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
11805b4aa86bSJames Feist                     return;
11815b4aa86bSJames Feist                 }
11825b4aa86bSJames Feist 
118323a21a1cSEd Tanous                 const std::string& path = subtreeLocal[0].first;
118423a21a1cSEd Tanous                 const std::string& owner = subtreeLocal[0].second[0].first;
118573df0db0SJames Feist                 crow::connections::systemBus->async_method_call(
118673df0db0SJames Feist                     [path, owner, self](
118723a21a1cSEd Tanous                         const boost::system::error_code ec2,
118873df0db0SJames Feist                         const boost::container::flat_map<
118973df0db0SJames Feist                             std::string, std::variant<std::vector<std::string>,
119073df0db0SJames Feist                                                       std::string>>& resp) {
119123a21a1cSEd Tanous                         if (ec2)
119273df0db0SJames Feist                         {
119373df0db0SJames Feist                             BMCWEB_LOG_ERROR << "GetPIDValues: Can't get "
119473df0db0SJames Feist                                                 "thermalModeIface "
119573df0db0SJames Feist                                              << path;
119673df0db0SJames Feist                             messages::internalError(self->asyncResp->res);
119773df0db0SJames Feist                             return;
119873df0db0SJames Feist                         }
1199271584abSEd Tanous                         const std::string* current = nullptr;
1200271584abSEd Tanous                         const std::vector<std::string>* supported = nullptr;
120173df0db0SJames Feist                         for (auto& [key, value] : resp)
120273df0db0SJames Feist                         {
120373df0db0SJames Feist                             if (key == "Current")
120473df0db0SJames Feist                             {
120573df0db0SJames Feist                                 current = std::get_if<std::string>(&value);
120673df0db0SJames Feist                                 if (current == nullptr)
120773df0db0SJames Feist                                 {
120873df0db0SJames Feist                                     BMCWEB_LOG_ERROR
120973df0db0SJames Feist                                         << "GetPIDValues: thermal mode "
121073df0db0SJames Feist                                            "iface invalid "
121173df0db0SJames Feist                                         << path;
121273df0db0SJames Feist                                     messages::internalError(
121373df0db0SJames Feist                                         self->asyncResp->res);
121473df0db0SJames Feist                                     return;
121573df0db0SJames Feist                                 }
121673df0db0SJames Feist                             }
121773df0db0SJames Feist                             if (key == "Supported")
121873df0db0SJames Feist                             {
121973df0db0SJames Feist                                 supported =
122073df0db0SJames Feist                                     std::get_if<std::vector<std::string>>(
122173df0db0SJames Feist                                         &value);
122273df0db0SJames Feist                                 if (supported == nullptr)
122373df0db0SJames Feist                                 {
122473df0db0SJames Feist                                     BMCWEB_LOG_ERROR
122573df0db0SJames Feist                                         << "GetPIDValues: thermal mode "
122673df0db0SJames Feist                                            "iface invalid"
122773df0db0SJames Feist                                         << path;
122873df0db0SJames Feist                                     messages::internalError(
122973df0db0SJames Feist                                         self->asyncResp->res);
123073df0db0SJames Feist                                     return;
123173df0db0SJames Feist                                 }
123273df0db0SJames Feist                             }
123373df0db0SJames Feist                         }
123473df0db0SJames Feist                         if (current == nullptr || supported == nullptr)
123573df0db0SJames Feist                         {
123673df0db0SJames Feist                             BMCWEB_LOG_ERROR << "GetPIDValues: thermal mode "
123773df0db0SJames Feist                                                 "iface invalid "
123873df0db0SJames Feist                                              << path;
123973df0db0SJames Feist                             messages::internalError(self->asyncResp->res);
124073df0db0SJames Feist                             return;
124173df0db0SJames Feist                         }
124273df0db0SJames Feist                         self->currentProfile = *current;
124373df0db0SJames Feist                         self->supportedProfiles = *supported;
124473df0db0SJames Feist                     },
124573df0db0SJames Feist                     owner, path, "org.freedesktop.DBus.Properties", "GetAll",
124673df0db0SJames Feist                     thermalModeIface);
124773df0db0SJames Feist             },
124873df0db0SJames Feist             "xyz.openbmc_project.ObjectMapper",
124973df0db0SJames Feist             "/xyz/openbmc_project/object_mapper",
125073df0db0SJames Feist             "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0,
125173df0db0SJames Feist             std::array<const char*, 1>{thermalModeIface});
125273df0db0SJames Feist     }
125373df0db0SJames Feist 
125473df0db0SJames Feist     ~GetPIDValues()
125573df0db0SJames Feist     {
125673df0db0SJames Feist         if (asyncResp->res.result() != boost::beast::http::status::ok)
125773df0db0SJames Feist         {
125873df0db0SJames Feist             return;
125973df0db0SJames Feist         }
12605b4aa86bSJames Feist         // create map of <connection, path to objMgr>>
126173df0db0SJames Feist         boost::container::flat_map<std::string, std::string> objectMgrPaths;
12626bce33bcSJames Feist         boost::container::flat_set<std::string> calledConnections;
12635b4aa86bSJames Feist         for (const auto& pathGroup : subtree)
12645b4aa86bSJames Feist         {
12655b4aa86bSJames Feist             for (const auto& connectionGroup : pathGroup.second)
12665b4aa86bSJames Feist             {
12676bce33bcSJames Feist                 auto findConnection =
12686bce33bcSJames Feist                     calledConnections.find(connectionGroup.first);
12696bce33bcSJames Feist                 if (findConnection != calledConnections.end())
12706bce33bcSJames Feist                 {
12716bce33bcSJames Feist                     break;
12726bce33bcSJames Feist                 }
127373df0db0SJames Feist                 for (const std::string& interface : connectionGroup.second)
12745b4aa86bSJames Feist                 {
12755b4aa86bSJames Feist                     if (interface == objectManagerIface)
12765b4aa86bSJames Feist                     {
127773df0db0SJames Feist                         objectMgrPaths[connectionGroup.first] = pathGroup.first;
12785b4aa86bSJames Feist                     }
12795b4aa86bSJames Feist                     // this list is alphabetical, so we
12805b4aa86bSJames Feist                     // should have found the objMgr by now
12815b4aa86bSJames Feist                     if (interface == pidConfigurationIface ||
1282b7a08d04SJames Feist                         interface == pidZoneConfigurationIface ||
1283b7a08d04SJames Feist                         interface == stepwiseConfigurationIface)
12845b4aa86bSJames Feist                     {
12855b4aa86bSJames Feist                         auto findObjMgr =
12865b4aa86bSJames Feist                             objectMgrPaths.find(connectionGroup.first);
12875b4aa86bSJames Feist                         if (findObjMgr == objectMgrPaths.end())
12885b4aa86bSJames Feist                         {
12895b4aa86bSJames Feist                             BMCWEB_LOG_DEBUG << connectionGroup.first
12905b4aa86bSJames Feist                                              << "Has no Object Manager";
12915b4aa86bSJames Feist                             continue;
12925b4aa86bSJames Feist                         }
12936bce33bcSJames Feist 
12946bce33bcSJames Feist                         calledConnections.insert(connectionGroup.first);
12956bce33bcSJames Feist 
129673df0db0SJames Feist                         asyncPopulatePid(findObjMgr->first, findObjMgr->second,
129773df0db0SJames Feist                                          currentProfile, supportedProfiles,
129873df0db0SJames Feist                                          asyncResp);
12995b4aa86bSJames Feist                         break;
13005b4aa86bSJames Feist                     }
13015b4aa86bSJames Feist                 }
13025b4aa86bSJames Feist             }
13035b4aa86bSJames Feist         }
130473df0db0SJames Feist     }
130573df0db0SJames Feist 
130673df0db0SJames Feist     std::vector<std::string> supportedProfiles;
130773df0db0SJames Feist     std::string currentProfile;
130873df0db0SJames Feist     crow::openbmc_mapper::GetSubTreeType subtree;
13098d1b46d7Szhanghch05     std::shared_ptr<bmcweb::AsyncResp> asyncResp;
131073df0db0SJames Feist };
131173df0db0SJames Feist 
131273df0db0SJames Feist struct SetPIDValues : std::enable_shared_from_this<SetPIDValues>
131373df0db0SJames Feist {
131473df0db0SJames Feist 
13158d1b46d7Szhanghch05     SetPIDValues(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn,
131673df0db0SJames Feist                  nlohmann::json& data) :
1317271584abSEd Tanous         asyncResp(asyncRespIn)
131873df0db0SJames Feist     {
131973df0db0SJames Feist 
132073df0db0SJames Feist         std::optional<nlohmann::json> pidControllers;
132173df0db0SJames Feist         std::optional<nlohmann::json> fanControllers;
132273df0db0SJames Feist         std::optional<nlohmann::json> fanZones;
132373df0db0SJames Feist         std::optional<nlohmann::json> stepwiseControllers;
132473df0db0SJames Feist 
132573df0db0SJames Feist         if (!redfish::json_util::readJson(
132673df0db0SJames Feist                 data, asyncResp->res, "PidControllers", pidControllers,
132773df0db0SJames Feist                 "FanControllers", fanControllers, "FanZones", fanZones,
132873df0db0SJames Feist                 "StepwiseControllers", stepwiseControllers, "Profile", profile))
132973df0db0SJames Feist         {
133071f52d96SEd Tanous             BMCWEB_LOG_ERROR
133171f52d96SEd Tanous                 << "Illegal Property "
133271f52d96SEd Tanous                 << data.dump(2, ' ', true,
133371f52d96SEd Tanous                              nlohmann::json::error_handler_t::replace);
133473df0db0SJames Feist             return;
133573df0db0SJames Feist         }
133673df0db0SJames Feist         configuration.emplace_back("PidControllers", std::move(pidControllers));
133773df0db0SJames Feist         configuration.emplace_back("FanControllers", std::move(fanControllers));
133873df0db0SJames Feist         configuration.emplace_back("FanZones", std::move(fanZones));
133973df0db0SJames Feist         configuration.emplace_back("StepwiseControllers",
134073df0db0SJames Feist                                    std::move(stepwiseControllers));
134173df0db0SJames Feist     }
134273df0db0SJames Feist     void run()
134373df0db0SJames Feist     {
134473df0db0SJames Feist         if (asyncResp->res.result() != boost::beast::http::status::ok)
134573df0db0SJames Feist         {
134673df0db0SJames Feist             return;
134773df0db0SJames Feist         }
134873df0db0SJames Feist 
134973df0db0SJames Feist         std::shared_ptr<SetPIDValues> self = shared_from_this();
135073df0db0SJames Feist 
135173df0db0SJames Feist         // todo(james): might make sense to do a mapper call here if this
135273df0db0SJames Feist         // interface gets more traction
135373df0db0SJames Feist         crow::connections::systemBus->async_method_call(
135473df0db0SJames Feist             [self](const boost::system::error_code ec,
1355271584abSEd Tanous                    dbus::utility::ManagedObjectType& mObj) {
135673df0db0SJames Feist                 if (ec)
135773df0db0SJames Feist                 {
135873df0db0SJames Feist                     BMCWEB_LOG_ERROR << "Error communicating to Entity Manager";
135973df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
136073df0db0SJames Feist                     return;
136173df0db0SJames Feist                 }
1362e69d9de2SJames Feist                 const std::array<const char*, 3> configurations = {
1363e69d9de2SJames Feist                     pidConfigurationIface, pidZoneConfigurationIface,
1364e69d9de2SJames Feist                     stepwiseConfigurationIface};
1365e69d9de2SJames Feist 
136614b0b8d5SJames Feist                 for (const auto& [path, object] : mObj)
1367e69d9de2SJames Feist                 {
136814b0b8d5SJames Feist                     for (const auto& [interface, _] : object)
1369e69d9de2SJames Feist                     {
1370e69d9de2SJames Feist                         if (std::find(configurations.begin(),
1371e69d9de2SJames Feist                                       configurations.end(),
1372e69d9de2SJames Feist                                       interface) != configurations.end())
1373e69d9de2SJames Feist                         {
137414b0b8d5SJames Feist                             self->objectCount++;
1375e69d9de2SJames Feist                             break;
1376e69d9de2SJames Feist                         }
1377e69d9de2SJames Feist                     }
1378e69d9de2SJames Feist                 }
1379271584abSEd Tanous                 self->managedObj = std::move(mObj);
138073df0db0SJames Feist             },
138173df0db0SJames Feist             "xyz.openbmc_project.EntityManager", "/", objectManagerIface,
138273df0db0SJames Feist             "GetManagedObjects");
138373df0db0SJames Feist 
138473df0db0SJames Feist         // at the same time get the profile information
138573df0db0SJames Feist         crow::connections::systemBus->async_method_call(
138673df0db0SJames Feist             [self](const boost::system::error_code ec,
138773df0db0SJames Feist                    const crow::openbmc_mapper::GetSubTreeType& subtree) {
138873df0db0SJames Feist                 if (ec || subtree.empty())
138973df0db0SJames Feist                 {
139073df0db0SJames Feist                     return;
139173df0db0SJames Feist                 }
139273df0db0SJames Feist                 if (subtree[0].second.empty())
139373df0db0SJames Feist                 {
139473df0db0SJames Feist                     // invalid mapper response, should never happen
139573df0db0SJames Feist                     BMCWEB_LOG_ERROR << "SetPIDValues: Mapper Error";
139673df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
139773df0db0SJames Feist                     return;
139873df0db0SJames Feist                 }
139973df0db0SJames Feist 
140073df0db0SJames Feist                 const std::string& path = subtree[0].first;
140173df0db0SJames Feist                 const std::string& owner = subtree[0].second[0].first;
140273df0db0SJames Feist                 crow::connections::systemBus->async_method_call(
140373df0db0SJames Feist                     [self, path, owner](
1404cb13a392SEd Tanous                         const boost::system::error_code ec2,
140573df0db0SJames Feist                         const boost::container::flat_map<
140673df0db0SJames Feist                             std::string, std::variant<std::vector<std::string>,
1407271584abSEd Tanous                                                       std::string>>& r) {
1408cb13a392SEd Tanous                         if (ec2)
140973df0db0SJames Feist                         {
141073df0db0SJames Feist                             BMCWEB_LOG_ERROR << "SetPIDValues: Can't get "
141173df0db0SJames Feist                                                 "thermalModeIface "
141273df0db0SJames Feist                                              << path;
141373df0db0SJames Feist                             messages::internalError(self->asyncResp->res);
141473df0db0SJames Feist                             return;
141573df0db0SJames Feist                         }
1416271584abSEd Tanous                         const std::string* current = nullptr;
1417271584abSEd Tanous                         const std::vector<std::string>* supported = nullptr;
1418271584abSEd Tanous                         for (auto& [key, value] : r)
141973df0db0SJames Feist                         {
142073df0db0SJames Feist                             if (key == "Current")
142173df0db0SJames Feist                             {
142273df0db0SJames Feist                                 current = std::get_if<std::string>(&value);
142373df0db0SJames Feist                                 if (current == nullptr)
142473df0db0SJames Feist                                 {
142573df0db0SJames Feist                                     BMCWEB_LOG_ERROR
142673df0db0SJames Feist                                         << "SetPIDValues: thermal mode "
142773df0db0SJames Feist                                            "iface invalid "
142873df0db0SJames Feist                                         << path;
142973df0db0SJames Feist                                     messages::internalError(
143073df0db0SJames Feist                                         self->asyncResp->res);
143173df0db0SJames Feist                                     return;
143273df0db0SJames Feist                                 }
143373df0db0SJames Feist                             }
143473df0db0SJames Feist                             if (key == "Supported")
143573df0db0SJames Feist                             {
143673df0db0SJames Feist                                 supported =
143773df0db0SJames Feist                                     std::get_if<std::vector<std::string>>(
143873df0db0SJames Feist                                         &value);
143973df0db0SJames Feist                                 if (supported == nullptr)
144073df0db0SJames Feist                                 {
144173df0db0SJames Feist                                     BMCWEB_LOG_ERROR
144273df0db0SJames Feist                                         << "SetPIDValues: thermal mode "
144373df0db0SJames Feist                                            "iface invalid"
144473df0db0SJames Feist                                         << path;
144573df0db0SJames Feist                                     messages::internalError(
144673df0db0SJames Feist                                         self->asyncResp->res);
144773df0db0SJames Feist                                     return;
144873df0db0SJames Feist                                 }
144973df0db0SJames Feist                             }
145073df0db0SJames Feist                         }
145173df0db0SJames Feist                         if (current == nullptr || supported == nullptr)
145273df0db0SJames Feist                         {
145373df0db0SJames Feist                             BMCWEB_LOG_ERROR << "SetPIDValues: thermal mode "
145473df0db0SJames Feist                                                 "iface invalid "
145573df0db0SJames Feist                                              << path;
145673df0db0SJames Feist                             messages::internalError(self->asyncResp->res);
145773df0db0SJames Feist                             return;
145873df0db0SJames Feist                         }
145973df0db0SJames Feist                         self->currentProfile = *current;
146073df0db0SJames Feist                         self->supportedProfiles = *supported;
146173df0db0SJames Feist                         self->profileConnection = owner;
146273df0db0SJames Feist                         self->profilePath = path;
146373df0db0SJames Feist                     },
146473df0db0SJames Feist                     owner, path, "org.freedesktop.DBus.Properties", "GetAll",
146573df0db0SJames Feist                     thermalModeIface);
14665b4aa86bSJames Feist             },
14675b4aa86bSJames Feist             "xyz.openbmc_project.ObjectMapper",
14685b4aa86bSJames Feist             "/xyz/openbmc_project/object_mapper",
14695b4aa86bSJames Feist             "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0,
147073df0db0SJames Feist             std::array<const char*, 1>{thermalModeIface});
147173df0db0SJames Feist     }
147273df0db0SJames Feist     ~SetPIDValues()
147373df0db0SJames Feist     {
147473df0db0SJames Feist         if (asyncResp->res.result() != boost::beast::http::status::ok)
147573df0db0SJames Feist         {
147673df0db0SJames Feist             return;
14775b4aa86bSJames Feist         }
14788d1b46d7Szhanghch05         std::shared_ptr<bmcweb::AsyncResp> response = asyncResp;
147973df0db0SJames Feist         if (profile)
148073df0db0SJames Feist         {
148173df0db0SJames Feist             if (std::find(supportedProfiles.begin(), supportedProfiles.end(),
148273df0db0SJames Feist                           *profile) == supportedProfiles.end())
148373df0db0SJames Feist             {
148473df0db0SJames Feist                 messages::actionParameterUnknown(response->res, "Profile",
148573df0db0SJames Feist                                                  *profile);
148673df0db0SJames Feist                 return;
148773df0db0SJames Feist             }
148873df0db0SJames Feist             currentProfile = *profile;
148973df0db0SJames Feist             crow::connections::systemBus->async_method_call(
149073df0db0SJames Feist                 [response](const boost::system::error_code ec) {
149173df0db0SJames Feist                     if (ec)
149273df0db0SJames Feist                     {
149373df0db0SJames Feist                         BMCWEB_LOG_ERROR << "Error patching profile" << ec;
149473df0db0SJames Feist                         messages::internalError(response->res);
149573df0db0SJames Feist                     }
149673df0db0SJames Feist                 },
149773df0db0SJames Feist                 profileConnection, profilePath,
149873df0db0SJames Feist                 "org.freedesktop.DBus.Properties", "Set", thermalModeIface,
149973df0db0SJames Feist                 "Current", std::variant<std::string>(*profile));
150073df0db0SJames Feist         }
150173df0db0SJames Feist 
150273df0db0SJames Feist         for (auto& containerPair : configuration)
150373df0db0SJames Feist         {
150473df0db0SJames Feist             auto& container = containerPair.second;
150573df0db0SJames Feist             if (!container)
150673df0db0SJames Feist             {
150773df0db0SJames Feist                 continue;
150873df0db0SJames Feist             }
15096ee7f774SJames Feist             BMCWEB_LOG_DEBUG << *container;
15106ee7f774SJames Feist 
151173df0db0SJames Feist             std::string& type = containerPair.first;
151273df0db0SJames Feist 
151373df0db0SJames Feist             for (nlohmann::json::iterator it = container->begin();
151417a897dfSManojkiran Eda                  it != container->end(); ++it)
151573df0db0SJames Feist             {
151673df0db0SJames Feist                 const auto& name = it.key();
15176ee7f774SJames Feist                 BMCWEB_LOG_DEBUG << "looking for " << name;
15186ee7f774SJames Feist 
151973df0db0SJames Feist                 auto pathItr =
152073df0db0SJames Feist                     std::find_if(managedObj.begin(), managedObj.end(),
152173df0db0SJames Feist                                  [&name](const auto& obj) {
152273df0db0SJames Feist                                      return boost::algorithm::ends_with(
152373df0db0SJames Feist                                          obj.first.str, "/" + name);
152473df0db0SJames Feist                                  });
152573df0db0SJames Feist                 boost::container::flat_map<std::string,
152673df0db0SJames Feist                                            dbus::utility::DbusVariantType>
152773df0db0SJames Feist                     output;
152873df0db0SJames Feist 
152973df0db0SJames Feist                 output.reserve(16); // The pid interface length
153073df0db0SJames Feist 
153173df0db0SJames Feist                 // determines if we're patching entity-manager or
153273df0db0SJames Feist                 // creating a new object
153373df0db0SJames Feist                 bool createNewObject = (pathItr == managedObj.end());
15346ee7f774SJames Feist                 BMCWEB_LOG_DEBUG << "Found = " << !createNewObject;
15356ee7f774SJames Feist 
153673df0db0SJames Feist                 std::string iface;
153773df0db0SJames Feist                 if (type == "PidControllers" || type == "FanControllers")
153873df0db0SJames Feist                 {
153973df0db0SJames Feist                     iface = pidConfigurationIface;
154073df0db0SJames Feist                     if (!createNewObject &&
154173df0db0SJames Feist                         pathItr->second.find(pidConfigurationIface) ==
154273df0db0SJames Feist                             pathItr->second.end())
154373df0db0SJames Feist                     {
154473df0db0SJames Feist                         createNewObject = true;
154573df0db0SJames Feist                     }
154673df0db0SJames Feist                 }
154773df0db0SJames Feist                 else if (type == "FanZones")
154873df0db0SJames Feist                 {
154973df0db0SJames Feist                     iface = pidZoneConfigurationIface;
155073df0db0SJames Feist                     if (!createNewObject &&
155173df0db0SJames Feist                         pathItr->second.find(pidZoneConfigurationIface) ==
155273df0db0SJames Feist                             pathItr->second.end())
155373df0db0SJames Feist                     {
155473df0db0SJames Feist 
155573df0db0SJames Feist                         createNewObject = true;
155673df0db0SJames Feist                     }
155773df0db0SJames Feist                 }
155873df0db0SJames Feist                 else if (type == "StepwiseControllers")
155973df0db0SJames Feist                 {
156073df0db0SJames Feist                     iface = stepwiseConfigurationIface;
156173df0db0SJames Feist                     if (!createNewObject &&
156273df0db0SJames Feist                         pathItr->second.find(stepwiseConfigurationIface) ==
156373df0db0SJames Feist                             pathItr->second.end())
156473df0db0SJames Feist                     {
156573df0db0SJames Feist                         createNewObject = true;
156673df0db0SJames Feist                     }
156773df0db0SJames Feist                 }
15686ee7f774SJames Feist 
15696ee7f774SJames Feist                 if (createNewObject && it.value() == nullptr)
15706ee7f774SJames Feist                 {
15714e0453b1SGunnar Mills                     // can't delete a non-existent object
15726ee7f774SJames Feist                     messages::invalidObject(response->res, name);
15736ee7f774SJames Feist                     continue;
15746ee7f774SJames Feist                 }
15756ee7f774SJames Feist 
15766ee7f774SJames Feist                 std::string path;
15776ee7f774SJames Feist                 if (pathItr != managedObj.end())
15786ee7f774SJames Feist                 {
15796ee7f774SJames Feist                     path = pathItr->first.str;
15806ee7f774SJames Feist                 }
15816ee7f774SJames Feist 
158273df0db0SJames Feist                 BMCWEB_LOG_DEBUG << "Create new = " << createNewObject << "\n";
1583e69d9de2SJames Feist 
1584e69d9de2SJames Feist                 // arbitrary limit to avoid attacks
1585e69d9de2SJames Feist                 constexpr const size_t controllerLimit = 500;
158614b0b8d5SJames Feist                 if (createNewObject && objectCount >= controllerLimit)
1587e69d9de2SJames Feist                 {
1588e69d9de2SJames Feist                     messages::resourceExhaustion(response->res, type);
1589e69d9de2SJames Feist                     continue;
1590e69d9de2SJames Feist                 }
1591e69d9de2SJames Feist 
159273df0db0SJames Feist                 output["Name"] = boost::replace_all_copy(name, "_", " ");
159373df0db0SJames Feist 
159473df0db0SJames Feist                 std::string chassis;
159573df0db0SJames Feist                 CreatePIDRet ret = createPidInterface(
15966ee7f774SJames Feist                     response, type, it, path, managedObj, createNewObject,
15976ee7f774SJames Feist                     output, chassis, currentProfile);
159873df0db0SJames Feist                 if (ret == CreatePIDRet::fail)
159973df0db0SJames Feist                 {
160073df0db0SJames Feist                     return;
160173df0db0SJames Feist                 }
16023174e4dfSEd Tanous                 if (ret == CreatePIDRet::del)
160373df0db0SJames Feist                 {
160473df0db0SJames Feist                     continue;
160573df0db0SJames Feist                 }
160673df0db0SJames Feist 
160773df0db0SJames Feist                 if (!createNewObject)
160873df0db0SJames Feist                 {
160973df0db0SJames Feist                     for (const auto& property : output)
161073df0db0SJames Feist                     {
161173df0db0SJames Feist                         crow::connections::systemBus->async_method_call(
161273df0db0SJames Feist                             [response,
161373df0db0SJames Feist                              propertyName{std::string(property.first)}](
161473df0db0SJames Feist                                 const boost::system::error_code ec) {
161573df0db0SJames Feist                                 if (ec)
161673df0db0SJames Feist                                 {
161773df0db0SJames Feist                                     BMCWEB_LOG_ERROR << "Error patching "
161873df0db0SJames Feist                                                      << propertyName << ": "
161973df0db0SJames Feist                                                      << ec;
162073df0db0SJames Feist                                     messages::internalError(response->res);
162173df0db0SJames Feist                                     return;
162273df0db0SJames Feist                                 }
162373df0db0SJames Feist                                 messages::success(response->res);
162473df0db0SJames Feist                             },
16256ee7f774SJames Feist                             "xyz.openbmc_project.EntityManager", path,
162673df0db0SJames Feist                             "org.freedesktop.DBus.Properties", "Set", iface,
162773df0db0SJames Feist                             property.first, property.second);
162873df0db0SJames Feist                     }
162973df0db0SJames Feist                 }
163073df0db0SJames Feist                 else
163173df0db0SJames Feist                 {
163273df0db0SJames Feist                     if (chassis.empty())
163373df0db0SJames Feist                     {
163473df0db0SJames Feist                         BMCWEB_LOG_ERROR << "Failed to get chassis from config";
163573df0db0SJames Feist                         messages::invalidObject(response->res, name);
163673df0db0SJames Feist                         return;
163773df0db0SJames Feist                     }
163873df0db0SJames Feist 
163973df0db0SJames Feist                     bool foundChassis = false;
164073df0db0SJames Feist                     for (const auto& obj : managedObj)
164173df0db0SJames Feist                     {
164273df0db0SJames Feist                         if (boost::algorithm::ends_with(obj.first.str, chassis))
164373df0db0SJames Feist                         {
164473df0db0SJames Feist                             chassis = obj.first.str;
164573df0db0SJames Feist                             foundChassis = true;
164673df0db0SJames Feist                             break;
164773df0db0SJames Feist                         }
164873df0db0SJames Feist                     }
164973df0db0SJames Feist                     if (!foundChassis)
165073df0db0SJames Feist                     {
165173df0db0SJames Feist                         BMCWEB_LOG_ERROR << "Failed to find chassis on dbus";
165273df0db0SJames Feist                         messages::resourceMissingAtURI(
165373df0db0SJames Feist                             response->res, "/redfish/v1/Chassis/" + chassis);
165473df0db0SJames Feist                         return;
165573df0db0SJames Feist                     }
165673df0db0SJames Feist 
165773df0db0SJames Feist                     crow::connections::systemBus->async_method_call(
165873df0db0SJames Feist                         [response](const boost::system::error_code ec) {
165973df0db0SJames Feist                             if (ec)
166073df0db0SJames Feist                             {
166173df0db0SJames Feist                                 BMCWEB_LOG_ERROR << "Error Adding Pid Object "
166273df0db0SJames Feist                                                  << ec;
166373df0db0SJames Feist                                 messages::internalError(response->res);
166473df0db0SJames Feist                                 return;
166573df0db0SJames Feist                             }
166673df0db0SJames Feist                             messages::success(response->res);
166773df0db0SJames Feist                         },
166873df0db0SJames Feist                         "xyz.openbmc_project.EntityManager", chassis,
166973df0db0SJames Feist                         "xyz.openbmc_project.AddObject", "AddObject", output);
167073df0db0SJames Feist                 }
167173df0db0SJames Feist             }
167273df0db0SJames Feist         }
167373df0db0SJames Feist     }
16748d1b46d7Szhanghch05     std::shared_ptr<bmcweb::AsyncResp> asyncResp;
167573df0db0SJames Feist     std::vector<std::pair<std::string, std::optional<nlohmann::json>>>
167673df0db0SJames Feist         configuration;
167773df0db0SJames Feist     std::optional<std::string> profile;
167873df0db0SJames Feist     dbus::utility::ManagedObjectType managedObj;
167973df0db0SJames Feist     std::vector<std::string> supportedProfiles;
168073df0db0SJames Feist     std::string currentProfile;
168173df0db0SJames Feist     std::string profileConnection;
168273df0db0SJames Feist     std::string profilePath;
168314b0b8d5SJames Feist     size_t objectCount = 0;
168473df0db0SJames Feist };
168573df0db0SJames Feist 
1686071d8fdfSSunnySrivastava1984 /**
1687071d8fdfSSunnySrivastava1984  * @brief Retrieves BMC manager location data over DBus
1688071d8fdfSSunnySrivastava1984  *
1689071d8fdfSSunnySrivastava1984  * @param[in] aResp Shared pointer for completing asynchronous calls
1690071d8fdfSSunnySrivastava1984  * @param[in] connectionName - service name
1691071d8fdfSSunnySrivastava1984  * @param[in] path - object path
1692071d8fdfSSunnySrivastava1984  * @return none
1693071d8fdfSSunnySrivastava1984  */
16948d1b46d7Szhanghch05 inline void getLocation(const std::shared_ptr<bmcweb::AsyncResp>& aResp,
1695071d8fdfSSunnySrivastava1984                         const std::string& connectionName,
1696071d8fdfSSunnySrivastava1984                         const std::string& path)
1697071d8fdfSSunnySrivastava1984 {
1698071d8fdfSSunnySrivastava1984     BMCWEB_LOG_DEBUG << "Get BMC manager Location data.";
1699071d8fdfSSunnySrivastava1984 
1700071d8fdfSSunnySrivastava1984     crow::connections::systemBus->async_method_call(
1701071d8fdfSSunnySrivastava1984         [aResp](const boost::system::error_code ec,
1702071d8fdfSSunnySrivastava1984                 const std::variant<std::string>& property) {
1703071d8fdfSSunnySrivastava1984             if (ec)
1704071d8fdfSSunnySrivastava1984             {
1705071d8fdfSSunnySrivastava1984                 BMCWEB_LOG_DEBUG << "DBUS response error for "
1706071d8fdfSSunnySrivastava1984                                     "Location";
1707071d8fdfSSunnySrivastava1984                 messages::internalError(aResp->res);
1708071d8fdfSSunnySrivastava1984                 return;
1709071d8fdfSSunnySrivastava1984             }
1710071d8fdfSSunnySrivastava1984 
1711071d8fdfSSunnySrivastava1984             const std::string* value = std::get_if<std::string>(&property);
1712071d8fdfSSunnySrivastava1984 
1713071d8fdfSSunnySrivastava1984             if (value == nullptr)
1714071d8fdfSSunnySrivastava1984             {
1715071d8fdfSSunnySrivastava1984                 // illegal value
1716071d8fdfSSunnySrivastava1984                 messages::internalError(aResp->res);
1717071d8fdfSSunnySrivastava1984                 return;
1718071d8fdfSSunnySrivastava1984             }
1719071d8fdfSSunnySrivastava1984 
1720071d8fdfSSunnySrivastava1984             aResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] =
1721071d8fdfSSunnySrivastava1984                 *value;
1722071d8fdfSSunnySrivastava1984         },
1723071d8fdfSSunnySrivastava1984         connectionName, path, "org.freedesktop.DBus.Properties", "Get",
1724071d8fdfSSunnySrivastava1984         "xyz.openbmc_project.Inventory.Decorator."
1725071d8fdfSSunnySrivastava1984         "LocationCode",
1726071d8fdfSSunnySrivastava1984         "LocationCode");
1727071d8fdfSSunnySrivastava1984 }
17287e860f15SJohn Edward Broadbent // avoid name collision systems.hpp
17297e860f15SJohn Edward Broadbent inline void
17307e860f15SJohn Edward Broadbent     managerGetLastResetTime(const std::shared_ptr<bmcweb::AsyncResp>& aResp)
17314bf2b033SGunnar Mills {
17324bf2b033SGunnar Mills     BMCWEB_LOG_DEBUG << "Getting Manager Last Reset Time";
17334bf2b033SGunnar Mills 
17344bf2b033SGunnar Mills     crow::connections::systemBus->async_method_call(
17354bf2b033SGunnar Mills         [aResp](const boost::system::error_code ec,
17364bf2b033SGunnar Mills                 std::variant<uint64_t>& lastResetTime) {
17374bf2b033SGunnar Mills             if (ec)
17384bf2b033SGunnar Mills             {
17394bf2b033SGunnar Mills                 BMCWEB_LOG_DEBUG << "D-BUS response error " << ec;
17404bf2b033SGunnar Mills                 return;
17414bf2b033SGunnar Mills             }
17424bf2b033SGunnar Mills 
17434bf2b033SGunnar Mills             const uint64_t* lastResetTimePtr =
17444bf2b033SGunnar Mills                 std::get_if<uint64_t>(&lastResetTime);
17454bf2b033SGunnar Mills 
17464bf2b033SGunnar Mills             if (!lastResetTimePtr)
17474bf2b033SGunnar Mills             {
17484bf2b033SGunnar Mills                 messages::internalError(aResp->res);
17494bf2b033SGunnar Mills                 return;
17504bf2b033SGunnar Mills             }
17514bf2b033SGunnar Mills             // LastRebootTime is epoch time, in milliseconds
17524bf2b033SGunnar Mills             // https://github.com/openbmc/phosphor-dbus-interfaces/blob/7f9a128eb9296e926422ddc312c148b625890bb6/xyz/openbmc_project/State/BMC.interface.yaml#L19
17534bf2b033SGunnar Mills             time_t lastResetTimeStamp =
17544bf2b033SGunnar Mills                 static_cast<time_t>(*lastResetTimePtr / 1000);
17554bf2b033SGunnar Mills 
17564bf2b033SGunnar Mills             // Convert to ISO 8601 standard
17574bf2b033SGunnar Mills             aResp->res.jsonValue["LastResetTime"] =
17584bf2b033SGunnar Mills                 crow::utility::getDateTime(lastResetTimeStamp);
17594bf2b033SGunnar Mills         },
17604bf2b033SGunnar Mills         "xyz.openbmc_project.State.BMC", "/xyz/openbmc_project/state/bmc0",
17614bf2b033SGunnar Mills         "org.freedesktop.DBus.Properties", "Get",
17624bf2b033SGunnar Mills         "xyz.openbmc_project.State.BMC", "LastRebootTime");
17634bf2b033SGunnar Mills }
17644bf2b033SGunnar Mills 
17654bfefa74SGunnar Mills /**
17664bfefa74SGunnar Mills  * @brief Set the running firmware image
17674bfefa74SGunnar Mills  *
17684bfefa74SGunnar Mills  * @param[i,o] aResp - Async response object
17694bfefa74SGunnar Mills  * @param[i] runningFirmwareTarget - Image to make the running image
17704bfefa74SGunnar Mills  *
17714bfefa74SGunnar Mills  * @return void
17724bfefa74SGunnar Mills  */
17737e860f15SJohn Edward Broadbent inline void
17747e860f15SJohn Edward Broadbent     setActiveFirmwareImage(const std::shared_ptr<bmcweb::AsyncResp>& aResp,
1775f23b7296SEd Tanous                            const std::string& runningFirmwareTarget)
17764bfefa74SGunnar Mills {
17774bfefa74SGunnar Mills     // Get the Id from /redfish/v1/UpdateService/FirmwareInventory/<Id>
1778f23b7296SEd Tanous     std::string::size_type idPos = runningFirmwareTarget.rfind('/');
17794bfefa74SGunnar Mills     if (idPos == std::string::npos)
17804bfefa74SGunnar Mills     {
17814bfefa74SGunnar Mills         messages::propertyValueNotInList(aResp->res, runningFirmwareTarget,
17824bfefa74SGunnar Mills                                          "@odata.id");
17834bfefa74SGunnar Mills         BMCWEB_LOG_DEBUG << "Can't parse firmware ID!";
17844bfefa74SGunnar Mills         return;
17854bfefa74SGunnar Mills     }
17864bfefa74SGunnar Mills     idPos++;
17874bfefa74SGunnar Mills     if (idPos >= runningFirmwareTarget.size())
17884bfefa74SGunnar Mills     {
17894bfefa74SGunnar Mills         messages::propertyValueNotInList(aResp->res, runningFirmwareTarget,
17904bfefa74SGunnar Mills                                          "@odata.id");
17914bfefa74SGunnar Mills         BMCWEB_LOG_DEBUG << "Invalid firmware ID.";
17924bfefa74SGunnar Mills         return;
17934bfefa74SGunnar Mills     }
17944bfefa74SGunnar Mills     std::string firmwareId = runningFirmwareTarget.substr(idPos);
17954bfefa74SGunnar Mills 
17964bfefa74SGunnar Mills     // Make sure the image is valid before setting priority
17974bfefa74SGunnar Mills     crow::connections::systemBus->async_method_call(
17987e860f15SJohn Edward Broadbent         [aResp, firmwareId, runningFirmwareTarget](
17997e860f15SJohn Edward Broadbent             const boost::system::error_code ec, ManagedObjectType& subtree) {
18004bfefa74SGunnar Mills             if (ec)
18014bfefa74SGunnar Mills             {
18024bfefa74SGunnar Mills                 BMCWEB_LOG_DEBUG << "D-Bus response error getting objects.";
18034bfefa74SGunnar Mills                 messages::internalError(aResp->res);
18044bfefa74SGunnar Mills                 return;
18054bfefa74SGunnar Mills             }
18064bfefa74SGunnar Mills 
18074bfefa74SGunnar Mills             if (subtree.size() == 0)
18084bfefa74SGunnar Mills             {
18094bfefa74SGunnar Mills                 BMCWEB_LOG_DEBUG << "Can't find image!";
18104bfefa74SGunnar Mills                 messages::internalError(aResp->res);
18114bfefa74SGunnar Mills                 return;
18124bfefa74SGunnar Mills             }
18134bfefa74SGunnar Mills 
18144bfefa74SGunnar Mills             bool foundImage = false;
18154bfefa74SGunnar Mills             for (auto& object : subtree)
18164bfefa74SGunnar Mills             {
18174bfefa74SGunnar Mills                 const std::string& path =
18184bfefa74SGunnar Mills                     static_cast<const std::string&>(object.first);
1819f23b7296SEd Tanous                 std::size_t idPos2 = path.rfind('/');
18204bfefa74SGunnar Mills 
18214bfefa74SGunnar Mills                 if (idPos2 == std::string::npos)
18224bfefa74SGunnar Mills                 {
18234bfefa74SGunnar Mills                     continue;
18244bfefa74SGunnar Mills                 }
18254bfefa74SGunnar Mills 
18264bfefa74SGunnar Mills                 idPos2++;
18274bfefa74SGunnar Mills                 if (idPos2 >= path.size())
18284bfefa74SGunnar Mills                 {
18294bfefa74SGunnar Mills                     continue;
18304bfefa74SGunnar Mills                 }
18314bfefa74SGunnar Mills 
18324bfefa74SGunnar Mills                 if (path.substr(idPos2) == firmwareId)
18334bfefa74SGunnar Mills                 {
18344bfefa74SGunnar Mills                     foundImage = true;
18354bfefa74SGunnar Mills                     break;
18364bfefa74SGunnar Mills                 }
18374bfefa74SGunnar Mills             }
18384bfefa74SGunnar Mills 
18394bfefa74SGunnar Mills             if (!foundImage)
18404bfefa74SGunnar Mills             {
18414bfefa74SGunnar Mills                 messages::propertyValueNotInList(
18424bfefa74SGunnar Mills                     aResp->res, runningFirmwareTarget, "@odata.id");
18434bfefa74SGunnar Mills                 BMCWEB_LOG_DEBUG << "Invalid firmware ID.";
18444bfefa74SGunnar Mills                 return;
18454bfefa74SGunnar Mills             }
18464bfefa74SGunnar Mills 
18477e860f15SJohn Edward Broadbent             BMCWEB_LOG_DEBUG
18487e860f15SJohn Edward Broadbent                 << "Setting firmware version " + firmwareId + " to priority 0.";
18494bfefa74SGunnar Mills 
18504bfefa74SGunnar Mills             // Only support Immediate
18514bfefa74SGunnar Mills             // An addition could be a Redfish Setting like
18524bfefa74SGunnar Mills             // ActiveSoftwareImageApplyTime and support OnReset
18534bfefa74SGunnar Mills             crow::connections::systemBus->async_method_call(
18544bfefa74SGunnar Mills                 [aResp](const boost::system::error_code ec) {
18554bfefa74SGunnar Mills                     if (ec)
18564bfefa74SGunnar Mills                     {
18574bfefa74SGunnar Mills                         BMCWEB_LOG_DEBUG << "D-Bus response error setting.";
18584bfefa74SGunnar Mills                         messages::internalError(aResp->res);
18594bfefa74SGunnar Mills                         return;
18604bfefa74SGunnar Mills                     }
18614bfefa74SGunnar Mills                     doBMCGracefulRestart(aResp);
18624bfefa74SGunnar Mills                 },
18634bfefa74SGunnar Mills 
18644bfefa74SGunnar Mills                 "xyz.openbmc_project.Software.BMC.Updater",
18654bfefa74SGunnar Mills                 "/xyz/openbmc_project/software/" + firmwareId,
18664bfefa74SGunnar Mills                 "org.freedesktop.DBus.Properties", "Set",
18677e860f15SJohn Edward Broadbent                 "xyz.openbmc_project.Software.RedundancyPriority", "Priority",
18687e860f15SJohn Edward Broadbent                 std::variant<uint8_t>(static_cast<uint8_t>(0)));
18694bfefa74SGunnar Mills         },
18704bfefa74SGunnar Mills         "xyz.openbmc_project.Software.BMC.Updater",
18717e860f15SJohn Edward Broadbent         "/xyz/openbmc_project/software", "org.freedesktop.DBus.ObjectManager",
18727e860f15SJohn Edward Broadbent         "GetManagedObjects");
18734bfefa74SGunnar Mills }
18744bfefa74SGunnar Mills 
18757e860f15SJohn Edward Broadbent inline void setDateTime(std::shared_ptr<bmcweb::AsyncResp> aResp,
18767e860f15SJohn Edward Broadbent                         std::string datetime)
1877af5d6058SSantosh Puranik {
1878af5d6058SSantosh Puranik     BMCWEB_LOG_DEBUG << "Set date time: " << datetime;
1879af5d6058SSantosh Puranik 
1880af5d6058SSantosh Puranik     std::stringstream stream(datetime);
1881af5d6058SSantosh Puranik     // Convert from ISO 8601 to boost local_time
1882af5d6058SSantosh Puranik     // (BMC only has time in UTC)
1883af5d6058SSantosh Puranik     boost::posix_time::ptime posixTime;
1884af5d6058SSantosh Puranik     boost::posix_time::ptime epoch(boost::gregorian::date(1970, 1, 1));
1885af5d6058SSantosh Puranik     // Facet gets deleted with the stringsteam
1886af5d6058SSantosh Puranik     auto ifc = std::make_unique<boost::local_time::local_time_input_facet>(
1887af5d6058SSantosh Puranik         "%Y-%m-%d %H:%M:%S%F %ZP");
1888af5d6058SSantosh Puranik     stream.imbue(std::locale(stream.getloc(), ifc.release()));
1889af5d6058SSantosh Puranik 
18907e860f15SJohn Edward Broadbent     boost::local_time::local_date_time ldt(boost::local_time::not_a_date_time);
1891af5d6058SSantosh Puranik 
1892af5d6058SSantosh Puranik     if (stream >> ldt)
1893af5d6058SSantosh Puranik     {
1894af5d6058SSantosh Puranik         posixTime = ldt.utc_time();
1895af5d6058SSantosh Puranik         boost::posix_time::time_duration dur = posixTime - epoch;
18967e860f15SJohn Edward Broadbent         uint64_t durMicroSecs = static_cast<uint64_t>(dur.total_microseconds());
1897af5d6058SSantosh Puranik         crow::connections::systemBus->async_method_call(
1898af5d6058SSantosh Puranik             [aResp{std::move(aResp)}, datetime{std::move(datetime)}](
1899af5d6058SSantosh Puranik                 const boost::system::error_code ec) {
1900af5d6058SSantosh Puranik                 if (ec)
1901af5d6058SSantosh Puranik                 {
1902af5d6058SSantosh Puranik                     BMCWEB_LOG_DEBUG << "Failed to set elapsed time. "
1903af5d6058SSantosh Puranik                                         "DBUS response error "
1904af5d6058SSantosh Puranik                                      << ec;
1905af5d6058SSantosh Puranik                     messages::internalError(aResp->res);
1906af5d6058SSantosh Puranik                     return;
1907af5d6058SSantosh Puranik                 }
1908af5d6058SSantosh Puranik                 aResp->res.jsonValue["DateTime"] = datetime;
1909af5d6058SSantosh Puranik             },
19107e860f15SJohn Edward Broadbent             "xyz.openbmc_project.Time.Manager", "/xyz/openbmc_project/time/bmc",
1911af5d6058SSantosh Puranik             "org.freedesktop.DBus.Properties", "Set",
1912af5d6058SSantosh Puranik             "xyz.openbmc_project.Time.EpochTime", "Elapsed",
1913af5d6058SSantosh Puranik             std::variant<uint64_t>(durMicroSecs));
1914af5d6058SSantosh Puranik     }
1915af5d6058SSantosh Puranik     else
1916af5d6058SSantosh Puranik     {
19177e860f15SJohn Edward Broadbent         messages::propertyValueFormatError(aResp->res, datetime, "DateTime");
1918af5d6058SSantosh Puranik         return;
1919af5d6058SSantosh Puranik     }
192083ff9ab6SJames Feist }
19219c310685SBorawski.Lukasz 
19227e860f15SJohn Edward Broadbent inline void requestRoutesManager(App& app)
19237e860f15SJohn Edward Broadbent {
19247e860f15SJohn Edward Broadbent     std::string uuid = persistent_data::getConfig().systemUuid;
19259c310685SBorawski.Lukasz 
19267e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/")
1927*ed398213SEd Tanous         .privileges(redfish::privileges::getManager)
19287e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::get)([uuid](const crow::Request&,
19297e860f15SJohn Edward Broadbent                                                        const std::shared_ptr<
19307e860f15SJohn Edward Broadbent                                                            bmcweb::AsyncResp>&
19317e860f15SJohn Edward Broadbent                                                            asyncResp) {
19327e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc";
19337e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["@odata.type"] =
19347e860f15SJohn Edward Broadbent                 "#Manager.v1_11_0.Manager";
19357e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["Id"] = "bmc";
19367e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["Name"] = "OpenBmc Manager";
19377e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["Description"] =
19387e860f15SJohn Edward Broadbent                 "Baseboard Management Controller";
19397e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["PowerState"] = "On";
19407e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["Status"] = {{"State", "Enabled"},
19417e860f15SJohn Edward Broadbent                                                   {"Health", "OK"}};
19427e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["ManagerType"] = "BMC";
19437e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["UUID"] = systemd_utils::getUuid();
19447e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["ServiceEntryPointUUID"] = uuid;
19457e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["Model"] =
19467e860f15SJohn Edward Broadbent                 "OpenBmc"; // TODO(ed), get model
19477e860f15SJohn Edward Broadbent 
19487e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["LogServices"] = {
19497e860f15SJohn Edward Broadbent                 {"@odata.id", "/redfish/v1/Managers/bmc/LogServices"}};
19507e860f15SJohn Edward Broadbent 
19517e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["NetworkProtocol"] = {
19527e860f15SJohn Edward Broadbent                 {"@odata.id", "/redfish/v1/Managers/bmc/NetworkProtocol"}};
19537e860f15SJohn Edward Broadbent 
19547e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["EthernetInterfaces"] = {
19557e860f15SJohn Edward Broadbent                 {"@odata.id", "/redfish/v1/Managers/bmc/EthernetInterfaces"}};
19567e860f15SJohn Edward Broadbent 
19577e860f15SJohn Edward Broadbent #ifdef BMCWEB_ENABLE_VM_NBDPROXY
19587e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["VirtualMedia"] = {
19597e860f15SJohn Edward Broadbent                 {"@odata.id", "/redfish/v1/Managers/bmc/VirtualMedia"}};
19607e860f15SJohn Edward Broadbent #endif // BMCWEB_ENABLE_VM_NBDPROXY
19617e860f15SJohn Edward Broadbent 
19627e860f15SJohn Edward Broadbent             // default oem data
19637e860f15SJohn Edward Broadbent             nlohmann::json& oem = asyncResp->res.jsonValue["Oem"];
19647e860f15SJohn Edward Broadbent             nlohmann::json& oemOpenbmc = oem["OpenBmc"];
19657e860f15SJohn Edward Broadbent             oem["@odata.type"] = "#OemManager.Oem";
19667e860f15SJohn Edward Broadbent             oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem";
19677e860f15SJohn Edward Broadbent             oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc";
19687e860f15SJohn Edward Broadbent             oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc";
19697e860f15SJohn Edward Broadbent             oemOpenbmc["Certificates"] = {
19707e860f15SJohn Edward Broadbent                 {"@odata.id",
19717e860f15SJohn Edward Broadbent                  "/redfish/v1/Managers/bmc/Truststore/Certificates"}};
19727e860f15SJohn Edward Broadbent 
19737e860f15SJohn Edward Broadbent             // Manager.Reset (an action) can be many values, OpenBMC only
19747e860f15SJohn Edward Broadbent             // supports BMC reboot.
19757e860f15SJohn Edward Broadbent             nlohmann::json& managerReset =
19767e860f15SJohn Edward Broadbent                 asyncResp->res.jsonValue["Actions"]["#Manager.Reset"];
19777e860f15SJohn Edward Broadbent             managerReset["target"] =
19787e860f15SJohn Edward Broadbent                 "/redfish/v1/Managers/bmc/Actions/Manager.Reset";
19797e860f15SJohn Edward Broadbent             managerReset["@Redfish.ActionInfo"] =
19807e860f15SJohn Edward Broadbent                 "/redfish/v1/Managers/bmc/ResetActionInfo";
19817e860f15SJohn Edward Broadbent 
19827e860f15SJohn Edward Broadbent             // ResetToDefaults (Factory Reset) has values like
19837e860f15SJohn Edward Broadbent             // PreserveNetworkAndUsers and PreserveNetwork that aren't supported
19847e860f15SJohn Edward Broadbent             // on OpenBMC
19857e860f15SJohn Edward Broadbent             nlohmann::json& resetToDefaults =
19867e860f15SJohn Edward Broadbent                 asyncResp->res.jsonValue["Actions"]["#Manager.ResetToDefaults"];
19877e860f15SJohn Edward Broadbent             resetToDefaults["target"] =
19887e860f15SJohn Edward Broadbent                 "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults";
19897e860f15SJohn Edward Broadbent             resetToDefaults["ResetType@Redfish.AllowableValues"] = {"ResetAll"};
19907e860f15SJohn Edward Broadbent 
19917c8c4058STejas Patil             std::pair<std::string, std::string> redfishDateTimeOffset =
19927c8c4058STejas Patil                 crow::utility::getDateTimeOffsetNow();
19937c8c4058STejas Patil 
19947c8c4058STejas Patil             asyncResp->res.jsonValue["DateTime"] = redfishDateTimeOffset.first;
19957c8c4058STejas Patil             asyncResp->res.jsonValue["DateTimeLocalOffset"] =
19967c8c4058STejas Patil                 redfishDateTimeOffset.second;
19977e860f15SJohn Edward Broadbent 
19980e8ac5e7SGunnar Mills             // TODO (Gunnar): Remove these one day since moved to ComputerSystem
19990e8ac5e7SGunnar Mills             // Still used by OCP profiles
20000e8ac5e7SGunnar Mills             // https://github.com/opencomputeproject/OCP-Profiles/issues/23
20017e860f15SJohn Edward Broadbent             // Fill in SerialConsole info
20027e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["SerialConsole"]["ServiceEnabled"] = true;
20037e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["SerialConsole"]["MaxConcurrentSessions"] =
20047e860f15SJohn Edward Broadbent                 15;
20057e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["SerialConsole"]["ConnectTypesSupported"] =
20067e860f15SJohn Edward Broadbent                 {"IPMI", "SSH"};
20077e860f15SJohn Edward Broadbent #ifdef BMCWEB_ENABLE_KVM
20087e860f15SJohn Edward Broadbent             // Fill in GraphicalConsole info
20097e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["GraphicalConsole"]["ServiceEnabled"] =
20107e860f15SJohn Edward Broadbent                 true;
20117e860f15SJohn Edward Broadbent             asyncResp->res
20127e860f15SJohn Edward Broadbent                 .jsonValue["GraphicalConsole"]["MaxConcurrentSessions"] = 4;
20137e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["GraphicalConsole"]
20147e860f15SJohn Edward Broadbent                                     ["ConnectTypesSupported"] = {"KVMIP"};
20157e860f15SJohn Edward Broadbent #endif // BMCWEB_ENABLE_KVM
20167e860f15SJohn Edward Broadbent 
20177e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["Links"]["ManagerForServers@odata.count"] =
20187e860f15SJohn Edward Broadbent                 1;
20197e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["Links"]["ManagerForServers"] = {
20207e860f15SJohn Edward Broadbent                 {{"@odata.id", "/redfish/v1/Systems/system"}}};
20217e860f15SJohn Edward Broadbent 
20227e860f15SJohn Edward Broadbent             auto health = std::make_shared<HealthPopulate>(asyncResp);
20237e860f15SJohn Edward Broadbent             health->isManagersHealth = true;
20247e860f15SJohn Edward Broadbent             health->populate();
20257e860f15SJohn Edward Broadbent 
20267e860f15SJohn Edward Broadbent             fw_util::populateFirmwareInformation(asyncResp, fw_util::bmcPurpose,
20277e860f15SJohn Edward Broadbent                                                  "FirmwareVersion", true);
20287e860f15SJohn Edward Broadbent 
20297e860f15SJohn Edward Broadbent             managerGetLastResetTime(asyncResp);
20307e860f15SJohn Edward Broadbent 
20317e860f15SJohn Edward Broadbent             auto pids = std::make_shared<GetPIDValues>(asyncResp);
20327e860f15SJohn Edward Broadbent             pids->run();
20337e860f15SJohn Edward Broadbent 
20347e860f15SJohn Edward Broadbent             getMainChassisId(
20357e860f15SJohn Edward Broadbent                 asyncResp, [](const std::string& chassisId,
20367e860f15SJohn Edward Broadbent                               const std::shared_ptr<bmcweb::AsyncResp>& aRsp) {
20377e860f15SJohn Edward Broadbent                     aRsp->res
20387e860f15SJohn Edward Broadbent                         .jsonValue["Links"]["ManagerForChassis@odata.count"] =
20397e860f15SJohn Edward Broadbent                         1;
20407e860f15SJohn Edward Broadbent                     aRsp->res.jsonValue["Links"]["ManagerForChassis"] = {
20417e860f15SJohn Edward Broadbent                         {{"@odata.id", "/redfish/v1/Chassis/" + chassisId}}};
20427e860f15SJohn Edward Broadbent                     aRsp->res.jsonValue["Links"]["ManagerInChassis"] = {
20437e860f15SJohn Edward Broadbent                         {"@odata.id", "/redfish/v1/Chassis/" + chassisId}};
20447e860f15SJohn Edward Broadbent                 });
20457e860f15SJohn Edward Broadbent 
20467e860f15SJohn Edward Broadbent             static bool started = false;
20477e860f15SJohn Edward Broadbent 
20487e860f15SJohn Edward Broadbent             if (!started)
20491abe55efSEd Tanous             {
20507e860f15SJohn Edward Broadbent                 crow::connections::systemBus->async_method_call(
20517e860f15SJohn Edward Broadbent                     [asyncResp](const boost::system::error_code ec,
20527e860f15SJohn Edward Broadbent                                 const std::variant<double>& resp) {
20537e860f15SJohn Edward Broadbent                         if (ec)
20541abe55efSEd Tanous                         {
20557e860f15SJohn Edward Broadbent                             BMCWEB_LOG_ERROR << "Error while getting progress";
20567e860f15SJohn Edward Broadbent                             messages::internalError(asyncResp->res);
20577e860f15SJohn Edward Broadbent                             return;
20587e860f15SJohn Edward Broadbent                         }
20597e860f15SJohn Edward Broadbent                         const double* val = std::get_if<double>(&resp);
20607e860f15SJohn Edward Broadbent                         if (val == nullptr)
20617e860f15SJohn Edward Broadbent                         {
20627e860f15SJohn Edward Broadbent                             BMCWEB_LOG_ERROR
20637e860f15SJohn Edward Broadbent                                 << "Invalid response while getting progress";
20647e860f15SJohn Edward Broadbent                             messages::internalError(asyncResp->res);
20657e860f15SJohn Edward Broadbent                             return;
20667e860f15SJohn Edward Broadbent                         }
20677e860f15SJohn Edward Broadbent                         if (*val < 1.0)
20687e860f15SJohn Edward Broadbent                         {
20697e860f15SJohn Edward Broadbent                             asyncResp->res.jsonValue["Status"]["State"] =
20707e860f15SJohn Edward Broadbent                                 "Starting";
20717e860f15SJohn Edward Broadbent                             started = true;
20727e860f15SJohn Edward Broadbent                         }
20737e860f15SJohn Edward Broadbent                     },
20747e860f15SJohn Edward Broadbent                     "org.freedesktop.systemd1", "/org/freedesktop/systemd1",
20757e860f15SJohn Edward Broadbent                     "org.freedesktop.DBus.Properties", "Get",
20767e860f15SJohn Edward Broadbent                     "org.freedesktop.systemd1.Manager", "Progress");
20779c310685SBorawski.Lukasz             }
20789c310685SBorawski.Lukasz 
20797e860f15SJohn Edward Broadbent             crow::connections::systemBus->async_method_call(
20807e860f15SJohn Edward Broadbent                 [asyncResp](
20817e860f15SJohn Edward Broadbent                     const boost::system::error_code ec,
20827e860f15SJohn Edward Broadbent                     const std::vector<
20837e860f15SJohn Edward Broadbent                         std::pair<std::string,
20847e860f15SJohn Edward Broadbent                                   std::vector<std::pair<
20857e860f15SJohn Edward Broadbent                                       std::string, std::vector<std::string>>>>>&
20867e860f15SJohn Edward Broadbent                         subtree) {
20877e860f15SJohn Edward Broadbent                     if (ec)
20881abe55efSEd Tanous                     {
20897e860f15SJohn Edward Broadbent                         BMCWEB_LOG_DEBUG
20907e860f15SJohn Edward Broadbent                             << "D-Bus response error on GetSubTree " << ec;
20917e860f15SJohn Edward Broadbent                         return;
20927e860f15SJohn Edward Broadbent                     }
20937e860f15SJohn Edward Broadbent                     if (subtree.size() == 0)
20947e860f15SJohn Edward Broadbent                     {
20957e860f15SJohn Edward Broadbent                         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                     {
21027e860f15SJohn Edward Broadbent                         BMCWEB_LOG_DEBUG
21037e860f15SJohn Edward Broadbent                             << "Found more than 1 bmc D-Bus object!";
21047e860f15SJohn Edward Broadbent                         messages::internalError(asyncResp->res);
21057e860f15SJohn Edward Broadbent                         return;
21067e860f15SJohn Edward Broadbent                     }
21077e860f15SJohn Edward Broadbent 
21087e860f15SJohn Edward Broadbent                     if (subtree[0].first.empty() ||
21097e860f15SJohn Edward Broadbent                         subtree[0].second.size() != 1)
21107e860f15SJohn Edward Broadbent                     {
21117e860f15SJohn Edward Broadbent                         BMCWEB_LOG_DEBUG << "Error getting bmc D-Bus object!";
21127e860f15SJohn Edward Broadbent                         messages::internalError(asyncResp->res);
21137e860f15SJohn Edward Broadbent                         return;
21147e860f15SJohn Edward Broadbent                     }
21157e860f15SJohn Edward Broadbent 
21167e860f15SJohn Edward Broadbent                     const std::string& path = subtree[0].first;
21177e860f15SJohn Edward Broadbent                     const std::string& connectionName =
21187e860f15SJohn Edward Broadbent                         subtree[0].second[0].first;
21197e860f15SJohn Edward Broadbent 
21207e860f15SJohn Edward Broadbent                     for (const auto& interfaceName :
21217e860f15SJohn Edward Broadbent                          subtree[0].second[0].second)
21227e860f15SJohn Edward Broadbent                     {
21237e860f15SJohn Edward Broadbent                         if (interfaceName ==
21247e860f15SJohn Edward Broadbent                             "xyz.openbmc_project.Inventory.Decorator.Asset")
21257e860f15SJohn Edward Broadbent                         {
21267e860f15SJohn Edward Broadbent                             crow::connections::systemBus->async_method_call(
21277e860f15SJohn Edward Broadbent                                 [asyncResp](
21287e860f15SJohn Edward Broadbent                                     const boost::system::error_code ec,
21297e860f15SJohn Edward Broadbent                                     const std::vector<
21307e860f15SJohn Edward Broadbent                                         std::pair<std::string,
21317e860f15SJohn Edward Broadbent                                                   std::variant<std::string>>>&
21327e860f15SJohn Edward Broadbent                                         propertiesList) {
21337e860f15SJohn Edward Broadbent                                     if (ec)
21347e860f15SJohn Edward Broadbent                                     {
21357e860f15SJohn Edward Broadbent                                         BMCWEB_LOG_DEBUG
21367e860f15SJohn Edward Broadbent                                             << "Can't get bmc asset!";
21377e860f15SJohn Edward Broadbent                                         return;
21387e860f15SJohn Edward Broadbent                                     }
21397e860f15SJohn Edward Broadbent                                     for (const std::pair<
21407e860f15SJohn Edward Broadbent                                              std::string,
21417e860f15SJohn Edward Broadbent                                              std::variant<std::string>>&
21427e860f15SJohn Edward Broadbent                                              property : propertiesList)
21437e860f15SJohn Edward Broadbent                                     {
21447e860f15SJohn Edward Broadbent                                         const std::string& propertyName =
21457e860f15SJohn Edward Broadbent                                             property.first;
21467e860f15SJohn Edward Broadbent 
21477e860f15SJohn Edward Broadbent                                         if ((propertyName == "PartNumber") ||
21487e860f15SJohn Edward Broadbent                                             (propertyName == "SerialNumber") ||
21497e860f15SJohn Edward Broadbent                                             (propertyName == "Manufacturer") ||
21507e860f15SJohn Edward Broadbent                                             (propertyName == "Model") ||
21517e860f15SJohn Edward Broadbent                                             (propertyName == "SparePartNumber"))
21527e860f15SJohn Edward Broadbent                                         {
21537e860f15SJohn Edward Broadbent                                             const std::string* value =
21547e860f15SJohn Edward Broadbent                                                 std::get_if<std::string>(
21557e860f15SJohn Edward Broadbent                                                     &property.second);
21567e860f15SJohn Edward Broadbent                                             if (value == nullptr)
21577e860f15SJohn Edward Broadbent                                             {
21587e860f15SJohn Edward Broadbent                                                 // illegal property
21597e860f15SJohn Edward Broadbent                                                 messages::internalError(
21607e860f15SJohn Edward Broadbent                                                     asyncResp->res);
21617e860f15SJohn Edward Broadbent                                                 return;
21627e860f15SJohn Edward Broadbent                                             }
21637e860f15SJohn Edward Broadbent                                             asyncResp->res
21647e860f15SJohn Edward Broadbent                                                 .jsonValue[propertyName] =
21657e860f15SJohn Edward Broadbent                                                 *value;
21667e860f15SJohn Edward Broadbent                                         }
21677e860f15SJohn Edward Broadbent                                     }
21687e860f15SJohn Edward Broadbent                                 },
21697e860f15SJohn Edward Broadbent                                 connectionName, path,
21707e860f15SJohn Edward Broadbent                                 "org.freedesktop.DBus.Properties", "GetAll",
21717e860f15SJohn Edward Broadbent                                 "xyz.openbmc_project.Inventory.Decorator."
21727e860f15SJohn Edward Broadbent                                 "Asset");
21737e860f15SJohn Edward Broadbent                         }
21747e860f15SJohn Edward Broadbent                         else if (interfaceName ==
21757e860f15SJohn Edward Broadbent                                  "xyz.openbmc_project.Inventory."
21767e860f15SJohn Edward Broadbent                                  "Decorator.LocationCode")
21777e860f15SJohn Edward Broadbent                         {
21787e860f15SJohn Edward Broadbent                             getLocation(asyncResp, connectionName, path);
21797e860f15SJohn Edward Broadbent                         }
21807e860f15SJohn Edward Broadbent                     }
21817e860f15SJohn Edward Broadbent                 },
21827e860f15SJohn Edward Broadbent                 "xyz.openbmc_project.ObjectMapper",
21837e860f15SJohn Edward Broadbent                 "/xyz/openbmc_project/object_mapper",
21847e860f15SJohn Edward Broadbent                 "xyz.openbmc_project.ObjectMapper", "GetSubTree",
21857e860f15SJohn Edward Broadbent                 "/xyz/openbmc_project/inventory", int32_t(0),
21867e860f15SJohn Edward Broadbent                 std::array<const char*, 1>{
21877e860f15SJohn Edward Broadbent                     "xyz.openbmc_project.Inventory.Item.Bmc"});
21887e860f15SJohn Edward Broadbent         });
21897e860f15SJohn Edward Broadbent 
21907e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/")
2191*ed398213SEd Tanous         .privileges(redfish::privileges::patchManager)
21927e860f15SJohn Edward Broadbent         .methods(
21937e860f15SJohn Edward Broadbent             boost::beast::http::verb::
21947e860f15SJohn Edward Broadbent                 patch)([](const crow::Request& req,
21957e860f15SJohn Edward Broadbent                           const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
21967e860f15SJohn Edward Broadbent             std::optional<nlohmann::json> oem;
21977e860f15SJohn Edward Broadbent             std::optional<nlohmann::json> links;
21987e860f15SJohn Edward Broadbent             std::optional<std::string> datetime;
21997e860f15SJohn Edward Broadbent 
22007e860f15SJohn Edward Broadbent             if (!json_util::readJson(req, asyncResp->res, "Oem", oem,
22017e860f15SJohn Edward Broadbent                                      "DateTime", datetime, "Links", links))
22027e860f15SJohn Edward Broadbent             {
22037e860f15SJohn Edward Broadbent                 return;
22047e860f15SJohn Edward Broadbent             }
22057e860f15SJohn Edward Broadbent 
22067e860f15SJohn Edward Broadbent             if (oem)
22077e860f15SJohn Edward Broadbent             {
22087e860f15SJohn Edward Broadbent                 std::optional<nlohmann::json> openbmc;
22097e860f15SJohn Edward Broadbent                 if (!redfish::json_util::readJson(*oem, asyncResp->res,
22107e860f15SJohn Edward Broadbent                                                   "OpenBmc", openbmc))
22117e860f15SJohn Edward Broadbent                 {
22127e860f15SJohn Edward Broadbent                     BMCWEB_LOG_ERROR
22137e860f15SJohn Edward Broadbent                         << "Illegal Property "
22147e860f15SJohn Edward Broadbent                         << oem->dump(2, ' ', true,
22157e860f15SJohn Edward Broadbent                                      nlohmann::json::error_handler_t::replace);
22167e860f15SJohn Edward Broadbent                     return;
22177e860f15SJohn Edward Broadbent                 }
22187e860f15SJohn Edward Broadbent                 if (openbmc)
22197e860f15SJohn Edward Broadbent                 {
22207e860f15SJohn Edward Broadbent                     std::optional<nlohmann::json> fan;
22217e860f15SJohn Edward Broadbent                     if (!redfish::json_util::readJson(*openbmc, asyncResp->res,
22227e860f15SJohn Edward Broadbent                                                       "Fan", fan))
22237e860f15SJohn Edward Broadbent                     {
22247e860f15SJohn Edward Broadbent                         BMCWEB_LOG_ERROR
22257e860f15SJohn Edward Broadbent                             << "Illegal Property "
22267e860f15SJohn Edward Broadbent                             << openbmc->dump(
22277e860f15SJohn Edward Broadbent                                    2, ' ', true,
22287e860f15SJohn Edward Broadbent                                    nlohmann::json::error_handler_t::replace);
22297e860f15SJohn Edward Broadbent                         return;
22307e860f15SJohn Edward Broadbent                     }
22317e860f15SJohn Edward Broadbent                     if (fan)
22327e860f15SJohn Edward Broadbent                     {
22337e860f15SJohn Edward Broadbent                         auto pid =
22347e860f15SJohn Edward Broadbent                             std::make_shared<SetPIDValues>(asyncResp, *fan);
22357e860f15SJohn Edward Broadbent                         pid->run();
22367e860f15SJohn Edward Broadbent                     }
22377e860f15SJohn Edward Broadbent                 }
22387e860f15SJohn Edward Broadbent             }
22397e860f15SJohn Edward Broadbent             if (links)
22407e860f15SJohn Edward Broadbent             {
22417e860f15SJohn Edward Broadbent                 std::optional<nlohmann::json> activeSoftwareImage;
22427e860f15SJohn Edward Broadbent                 if (!redfish::json_util::readJson(*links, asyncResp->res,
22437e860f15SJohn Edward Broadbent                                                   "ActiveSoftwareImage",
22447e860f15SJohn Edward Broadbent                                                   activeSoftwareImage))
22457e860f15SJohn Edward Broadbent                 {
22467e860f15SJohn Edward Broadbent                     return;
22477e860f15SJohn Edward Broadbent                 }
22487e860f15SJohn Edward Broadbent                 if (activeSoftwareImage)
22497e860f15SJohn Edward Broadbent                 {
22507e860f15SJohn Edward Broadbent                     std::optional<std::string> odataId;
22517e860f15SJohn Edward Broadbent                     if (!json_util::readJson(*activeSoftwareImage,
22527e860f15SJohn Edward Broadbent                                              asyncResp->res, "@odata.id",
22537e860f15SJohn Edward Broadbent                                              odataId))
22547e860f15SJohn Edward Broadbent                     {
22557e860f15SJohn Edward Broadbent                         return;
22567e860f15SJohn Edward Broadbent                     }
22577e860f15SJohn Edward Broadbent 
22587e860f15SJohn Edward Broadbent                     if (odataId)
22597e860f15SJohn Edward Broadbent                     {
22607e860f15SJohn Edward Broadbent                         setActiveFirmwareImage(asyncResp, *odataId);
22617e860f15SJohn Edward Broadbent                     }
22627e860f15SJohn Edward Broadbent                 }
22637e860f15SJohn Edward Broadbent             }
22647e860f15SJohn Edward Broadbent             if (datetime)
22657e860f15SJohn Edward Broadbent             {
22667e860f15SJohn Edward Broadbent                 setDateTime(asyncResp, std::move(*datetime));
22677e860f15SJohn Edward Broadbent             }
22687e860f15SJohn Edward Broadbent         });
22697e860f15SJohn Edward Broadbent }
22707e860f15SJohn Edward Broadbent 
22717e860f15SJohn Edward Broadbent inline void requestRoutesManagerCollection(App& app)
22727e860f15SJohn Edward Broadbent {
22737e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/Managers/")
2274*ed398213SEd Tanous         .privileges(redfish::privileges::getManagerCollection)
22757e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::get)(
22767e860f15SJohn Edward Broadbent             [](const crow::Request&,
22777e860f15SJohn Edward Broadbent                const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
227883ff9ab6SJames Feist                 // Collections don't include the static data added by SubRoute
227983ff9ab6SJames Feist                 // because it has a duplicate entry for members
22808d1b46d7Szhanghch05                 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers";
22818d1b46d7Szhanghch05                 asyncResp->res.jsonValue["@odata.type"] =
22828d1b46d7Szhanghch05                     "#ManagerCollection.ManagerCollection";
22838d1b46d7Szhanghch05                 asyncResp->res.jsonValue["Name"] = "Manager Collection";
22848d1b46d7Szhanghch05                 asyncResp->res.jsonValue["Members@odata.count"] = 1;
22858d1b46d7Szhanghch05                 asyncResp->res.jsonValue["Members"] = {
22865b4aa86bSJames Feist                     {{"@odata.id", "/redfish/v1/Managers/bmc"}}};
22877e860f15SJohn Edward Broadbent             });
22889c310685SBorawski.Lukasz }
22899c310685SBorawski.Lukasz } // namespace redfish
2290