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