xref: /openbmc/bmcweb/features/redfish/lib/managers.hpp (revision abf2add6a35aaf42ba60c4ae955a4d8925b13b19)
19c310685SBorawski.Lukasz /*
29c310685SBorawski.Lukasz // Copyright (c) 2018 Intel Corporation
39c310685SBorawski.Lukasz //
49c310685SBorawski.Lukasz // Licensed under the Apache License, Version 2.0 (the "License");
59c310685SBorawski.Lukasz // you may not use this file except in compliance with the License.
69c310685SBorawski.Lukasz // You may obtain a copy of the License at
79c310685SBorawski.Lukasz //
89c310685SBorawski.Lukasz //      http://www.apache.org/licenses/LICENSE-2.0
99c310685SBorawski.Lukasz //
109c310685SBorawski.Lukasz // Unless required by applicable law or agreed to in writing, software
119c310685SBorawski.Lukasz // distributed under the License is distributed on an "AS IS" BASIS,
129c310685SBorawski.Lukasz // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139c310685SBorawski.Lukasz // See the License for the specific language governing permissions and
149c310685SBorawski.Lukasz // limitations under the License.
159c310685SBorawski.Lukasz */
169c310685SBorawski.Lukasz #pragma once
179c310685SBorawski.Lukasz 
189c310685SBorawski.Lukasz #include "node.hpp"
199c310685SBorawski.Lukasz 
205b4aa86bSJames Feist #include <boost/algorithm/string/replace.hpp>
215b4aa86bSJames Feist #include <dbus_utility.hpp>
22*abf2add6SEd Tanous #include <variant>
235b4aa86bSJames Feist 
241abe55efSEd Tanous namespace redfish
251abe55efSEd Tanous {
26ed5befbdSJennifer Lee 
27ed5befbdSJennifer Lee /**
28ed5befbdSJennifer Lee  * ManagerActionsReset class supports handle POST method for Reset action.
29ed5befbdSJennifer Lee  * The class retrieves and sends data directly to dbus.
30ed5befbdSJennifer Lee  */
31ed5befbdSJennifer Lee class ManagerActionsReset : public Node
32ed5befbdSJennifer Lee {
33ed5befbdSJennifer Lee   public:
34ed5befbdSJennifer Lee     ManagerActionsReset(CrowApp& app) :
35ed5befbdSJennifer Lee         Node(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/")
36ed5befbdSJennifer Lee     {
37ed5befbdSJennifer Lee         entityPrivileges = {
38ed5befbdSJennifer Lee             {boost::beast::http::verb::get, {{"Login"}}},
39ed5befbdSJennifer Lee             {boost::beast::http::verb::head, {{"Login"}}},
40ed5befbdSJennifer Lee             {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
41ed5befbdSJennifer Lee             {boost::beast::http::verb::put, {{"ConfigureManager"}}},
42ed5befbdSJennifer Lee             {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
43ed5befbdSJennifer Lee             {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
44ed5befbdSJennifer Lee     }
45ed5befbdSJennifer Lee 
46ed5befbdSJennifer Lee   private:
47ed5befbdSJennifer Lee     /**
48ed5befbdSJennifer Lee      * Function handles POST method request.
49ed5befbdSJennifer Lee      * Analyzes POST body message before sends Reset request data to dbus.
50ed5befbdSJennifer Lee      * OpenBMC allows for ResetType is GracefulRestart only.
51ed5befbdSJennifer Lee      */
52ed5befbdSJennifer Lee     void doPost(crow::Response& res, const crow::Request& req,
53ed5befbdSJennifer Lee                 const std::vector<std::string>& params) override
54ed5befbdSJennifer Lee     {
55ed5befbdSJennifer Lee         std::string resetType;
56ed5befbdSJennifer Lee 
57ed5befbdSJennifer Lee         if (!json_util::readJson(req, res, "ResetType", resetType))
58ed5befbdSJennifer Lee         {
59ed5befbdSJennifer Lee             return;
60ed5befbdSJennifer Lee         }
61ed5befbdSJennifer Lee 
62ed5befbdSJennifer Lee         if (resetType != "GracefulRestart")
63ed5befbdSJennifer Lee         {
64ed5befbdSJennifer Lee             res.result(boost::beast::http::status::bad_request);
65ed5befbdSJennifer Lee             messages::actionParameterNotSupported(res, resetType, "ResetType");
66ed5befbdSJennifer Lee             BMCWEB_LOG_ERROR << "Request incorrect action parameter: "
67ed5befbdSJennifer Lee                              << resetType;
68ed5befbdSJennifer Lee             res.end();
69ed5befbdSJennifer Lee             return;
70ed5befbdSJennifer Lee         }
71ed5befbdSJennifer Lee         doBMCGracefulRestart(res, req, params);
72ed5befbdSJennifer Lee     }
73ed5befbdSJennifer Lee 
74ed5befbdSJennifer Lee     /**
75ed5befbdSJennifer Lee      * Function transceives data with dbus directly.
76ed5befbdSJennifer Lee      * All BMC state properties will be retrieved before sending reset request.
77ed5befbdSJennifer Lee      */
78ed5befbdSJennifer Lee     void doBMCGracefulRestart(crow::Response& res, const crow::Request& req,
79ed5befbdSJennifer Lee                               const std::vector<std::string>& params)
80ed5befbdSJennifer Lee     {
81ed5befbdSJennifer Lee         const char* processName = "xyz.openbmc_project.State.BMC";
82ed5befbdSJennifer Lee         const char* objectPath = "/xyz/openbmc_project/state/bmc0";
83ed5befbdSJennifer Lee         const char* interfaceName = "xyz.openbmc_project.State.BMC";
84ed5befbdSJennifer Lee         const std::string& propertyValue =
85ed5befbdSJennifer Lee             "xyz.openbmc_project.State.BMC.Transition.Reboot";
86ed5befbdSJennifer Lee         const char* destProperty = "RequestedBMCTransition";
87ed5befbdSJennifer Lee 
88ed5befbdSJennifer Lee         // Create the D-Bus variant for D-Bus call.
89ed5befbdSJennifer Lee         VariantType dbusPropertyValue(propertyValue);
90ed5befbdSJennifer Lee 
91ed5befbdSJennifer Lee         std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res);
92ed5befbdSJennifer Lee 
93ed5befbdSJennifer Lee         crow::connections::systemBus->async_method_call(
94ed5befbdSJennifer Lee             [asyncResp](const boost::system::error_code ec) {
95ed5befbdSJennifer Lee                 // Use "Set" method to set the property value.
96ed5befbdSJennifer Lee                 if (ec)
97ed5befbdSJennifer Lee                 {
98ed5befbdSJennifer Lee                     BMCWEB_LOG_ERROR << "[Set] Bad D-Bus request error: " << ec;
99ed5befbdSJennifer Lee                     messages::internalError(asyncResp->res);
100ed5befbdSJennifer Lee                     return;
101ed5befbdSJennifer Lee                 }
102ed5befbdSJennifer Lee 
103ed5befbdSJennifer Lee                 messages::success(asyncResp->res);
104ed5befbdSJennifer Lee             },
105ed5befbdSJennifer Lee             processName, objectPath, "org.freedesktop.DBus.Properties", "Set",
106ed5befbdSJennifer Lee             interfaceName, destProperty, dbusPropertyValue);
107ed5befbdSJennifer Lee     }
108ed5befbdSJennifer Lee };
109ed5befbdSJennifer Lee 
1105b4aa86bSJames Feist static constexpr const char* objectManagerIface =
1115b4aa86bSJames Feist     "org.freedesktop.DBus.ObjectManager";
1125b4aa86bSJames Feist static constexpr const char* pidConfigurationIface =
1135b4aa86bSJames Feist     "xyz.openbmc_project.Configuration.Pid";
1145b4aa86bSJames Feist static constexpr const char* pidZoneConfigurationIface =
1155b4aa86bSJames Feist     "xyz.openbmc_project.Configuration.Pid.Zone";
116b7a08d04SJames Feist static constexpr const char* stepwiseConfigurationIface =
117b7a08d04SJames Feist     "xyz.openbmc_project.Configuration.Stepwise";
1189c310685SBorawski.Lukasz 
1195b4aa86bSJames Feist static void asyncPopulatePid(const std::string& connection,
1205b4aa86bSJames Feist                              const std::string& path,
1215b4aa86bSJames Feist                              std::shared_ptr<AsyncResp> asyncResp)
1225b4aa86bSJames Feist {
1235b4aa86bSJames Feist 
1245b4aa86bSJames Feist     crow::connections::systemBus->async_method_call(
1255b4aa86bSJames Feist         [asyncResp](const boost::system::error_code ec,
1265b4aa86bSJames Feist                     const dbus::utility::ManagedObjectType& managedObj) {
1275b4aa86bSJames Feist             if (ec)
1285b4aa86bSJames Feist             {
1295b4aa86bSJames Feist                 BMCWEB_LOG_ERROR << ec;
1305b4aa86bSJames Feist                 asyncResp->res.jsonValue.clear();
131f12894f8SJason M. Bills                 messages::internalError(asyncResp->res);
1325b4aa86bSJames Feist                 return;
1335b4aa86bSJames Feist             }
1345b4aa86bSJames Feist             nlohmann::json& configRoot =
1355b4aa86bSJames Feist                 asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"];
1365b4aa86bSJames Feist             nlohmann::json& fans = configRoot["FanControllers"];
1375b4aa86bSJames Feist             fans["@odata.type"] = "#OemManager.FanControllers";
1385b4aa86bSJames Feist             fans["@odata.context"] =
1395b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.FanControllers";
1405b4aa86bSJames Feist             fans["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/"
1415b4aa86bSJames Feist                                 "Fan/FanControllers";
1425b4aa86bSJames Feist 
1435b4aa86bSJames Feist             nlohmann::json& pids = configRoot["PidControllers"];
1445b4aa86bSJames Feist             pids["@odata.type"] = "#OemManager.PidControllers";
1455b4aa86bSJames Feist             pids["@odata.context"] =
1465b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.PidControllers";
1475b4aa86bSJames Feist             pids["@odata.id"] =
1485b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers";
1495b4aa86bSJames Feist 
150b7a08d04SJames Feist             nlohmann::json& stepwise = configRoot["StepwiseControllers"];
151b7a08d04SJames Feist             stepwise["@odata.type"] = "#OemManager.StepwiseControllers";
152b7a08d04SJames Feist             stepwise["@odata.context"] =
153b7a08d04SJames Feist                 "/redfish/v1/$metadata#OemManager.StepwiseControllers";
154b7a08d04SJames Feist             stepwise["@odata.id"] =
155b7a08d04SJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers";
156b7a08d04SJames Feist 
1575b4aa86bSJames Feist             nlohmann::json& zones = configRoot["FanZones"];
1585b4aa86bSJames Feist             zones["@odata.id"] =
1595b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones";
1605b4aa86bSJames Feist             zones["@odata.type"] = "#OemManager.FanZones";
1615b4aa86bSJames Feist             zones["@odata.context"] =
1625b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.FanZones";
1635b4aa86bSJames Feist             configRoot["@odata.id"] =
1645b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan";
1655b4aa86bSJames Feist             configRoot["@odata.type"] = "#OemManager.Fan";
1665b4aa86bSJames Feist             configRoot["@odata.context"] =
1675b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.Fan";
1685b4aa86bSJames Feist 
1695b4aa86bSJames Feist             bool propertyError = false;
1705b4aa86bSJames Feist             for (const auto& pathPair : managedObj)
1715b4aa86bSJames Feist             {
1725b4aa86bSJames Feist                 for (const auto& intfPair : pathPair.second)
1735b4aa86bSJames Feist                 {
1745b4aa86bSJames Feist                     if (intfPair.first != pidConfigurationIface &&
175b7a08d04SJames Feist                         intfPair.first != pidZoneConfigurationIface &&
176b7a08d04SJames Feist                         intfPair.first != stepwiseConfigurationIface)
1775b4aa86bSJames Feist                     {
1785b4aa86bSJames Feist                         continue;
1795b4aa86bSJames Feist                     }
1805b4aa86bSJames Feist                     auto findName = intfPair.second.find("Name");
1815b4aa86bSJames Feist                     if (findName == intfPair.second.end())
1825b4aa86bSJames Feist                     {
1835b4aa86bSJames Feist                         BMCWEB_LOG_ERROR << "Pid Field missing Name";
184a08b46ccSJason M. Bills                         messages::internalError(asyncResp->res);
1855b4aa86bSJames Feist                         return;
1865b4aa86bSJames Feist                     }
1875b4aa86bSJames Feist                     const std::string* namePtr =
188*abf2add6SEd Tanous                         std::get_if<std::string>(&findName->second);
1895b4aa86bSJames Feist                     if (namePtr == nullptr)
1905b4aa86bSJames Feist                     {
1915b4aa86bSJames Feist                         BMCWEB_LOG_ERROR << "Pid Name Field illegal";
192b7a08d04SJames Feist                         messages::internalError(asyncResp->res);
1935b4aa86bSJames Feist                         return;
1945b4aa86bSJames Feist                     }
1955b4aa86bSJames Feist 
1965b4aa86bSJames Feist                     std::string name = *namePtr;
1975b4aa86bSJames Feist                     dbus::utility::escapePathForDbus(name);
198b7a08d04SJames Feist                     nlohmann::json* config = nullptr;
1995b4aa86bSJames Feist                     if (intfPair.first == pidZoneConfigurationIface)
2005b4aa86bSJames Feist                     {
2015b4aa86bSJames Feist                         std::string chassis;
2025b4aa86bSJames Feist                         if (!dbus::utility::getNthStringFromPath(
2035b4aa86bSJames Feist                                 pathPair.first.str, 5, chassis))
2045b4aa86bSJames Feist                         {
2055b4aa86bSJames Feist                             chassis = "#IllegalValue";
2065b4aa86bSJames Feist                         }
2075b4aa86bSJames Feist                         nlohmann::json& zone = zones[name];
2085b4aa86bSJames Feist                         zone["Chassis"] = {
2095b4aa86bSJames Feist                             {"@odata.id", "/redfish/v1/Chassis/" + chassis}};
2105b4aa86bSJames Feist                         zone["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/"
2115b4aa86bSJames Feist                                             "OpenBmc/Fan/FanZones/" +
2125b4aa86bSJames Feist                                             name;
2135b4aa86bSJames Feist                         zone["@odata.type"] = "#OemManager.FanZone";
2145b4aa86bSJames Feist                         zone["@odata.context"] =
2155b4aa86bSJames Feist                             "/redfish/v1/$metadata#OemManager.FanZone";
216b7a08d04SJames Feist                         config = &zone;
2175b4aa86bSJames Feist                     }
2185b4aa86bSJames Feist 
219b7a08d04SJames Feist                     else if (intfPair.first == stepwiseConfigurationIface)
2205b4aa86bSJames Feist                     {
221b7a08d04SJames Feist                         nlohmann::json& controller = stepwise[name];
222b7a08d04SJames Feist                         config = &controller;
2235b4aa86bSJames Feist 
224b7a08d04SJames Feist                         controller["@odata.id"] =
225b7a08d04SJames Feist                             "/redfish/v1/Managers/bmc#/Oem/"
226b7a08d04SJames Feist                             "OpenBmc/Fan/StepwiseControllers/" +
227b7a08d04SJames Feist                             std::string(name);
228b7a08d04SJames Feist                         controller["@odata.type"] =
229b7a08d04SJames Feist                             "#OemManager.StepwiseController";
230b7a08d04SJames Feist 
231b7a08d04SJames Feist                         controller["@odata.context"] =
232b7a08d04SJames Feist                             "/redfish/v1/"
233b7a08d04SJames Feist                             "$metadata#OemManager.StepwiseController";
2345b4aa86bSJames Feist                     }
2355b4aa86bSJames Feist 
2365b4aa86bSJames Feist                     // pid and fans are off the same configuration
237b7a08d04SJames Feist                     else if (intfPair.first == pidConfigurationIface)
2385b4aa86bSJames Feist                     {
2395b4aa86bSJames Feist                         const std::string* classPtr = nullptr;
2405b4aa86bSJames Feist                         auto findClass = intfPair.second.find("Class");
2415b4aa86bSJames Feist                         if (findClass != intfPair.second.end())
2425b4aa86bSJames Feist                         {
243*abf2add6SEd Tanous                             classPtr =
244*abf2add6SEd Tanous                                 std::get_if<std::string>(&findClass->second);
2455b4aa86bSJames Feist                         }
2465b4aa86bSJames Feist                         if (classPtr == nullptr)
2475b4aa86bSJames Feist                         {
2485b4aa86bSJames Feist                             BMCWEB_LOG_ERROR << "Pid Class Field illegal";
249a08b46ccSJason M. Bills                             messages::internalError(asyncResp->res);
2505b4aa86bSJames Feist                             return;
2515b4aa86bSJames Feist                         }
2525b4aa86bSJames Feist                         bool isFan = *classPtr == "fan";
2535b4aa86bSJames Feist                         nlohmann::json& element =
2545b4aa86bSJames Feist                             isFan ? fans[name] : pids[name];
255b7a08d04SJames Feist                         config = &element;
2565b4aa86bSJames Feist                         if (isFan)
2575b4aa86bSJames Feist                         {
2585b4aa86bSJames Feist                             element["@odata.id"] =
2595b4aa86bSJames Feist                                 "/redfish/v1/Managers/bmc#/Oem/"
2605b4aa86bSJames Feist                                 "OpenBmc/Fan/FanControllers/" +
2615b4aa86bSJames Feist                                 std::string(name);
2625b4aa86bSJames Feist                             element["@odata.type"] =
2635b4aa86bSJames Feist                                 "#OemManager.FanController";
2645b4aa86bSJames Feist 
2655b4aa86bSJames Feist                             element["@odata.context"] =
2665b4aa86bSJames Feist                                 "/redfish/v1/"
2675b4aa86bSJames Feist                                 "$metadata#OemManager.FanController";
2685b4aa86bSJames Feist                         }
2695b4aa86bSJames Feist                         else
2705b4aa86bSJames Feist                         {
2715b4aa86bSJames Feist                             element["@odata.id"] =
2725b4aa86bSJames Feist                                 "/redfish/v1/Managers/bmc#/Oem/"
2735b4aa86bSJames Feist                                 "OpenBmc/Fan/PidControllers/" +
2745b4aa86bSJames Feist                                 std::string(name);
2755b4aa86bSJames Feist                             element["@odata.type"] =
2765b4aa86bSJames Feist                                 "#OemManager.PidController";
2775b4aa86bSJames Feist                             element["@odata.context"] =
2785b4aa86bSJames Feist                                 "/redfish/v1/$metadata"
2795b4aa86bSJames Feist                                 "#OemManager.PidController";
2805b4aa86bSJames Feist                         }
281b7a08d04SJames Feist                     }
282b7a08d04SJames Feist                     else
283b7a08d04SJames Feist                     {
284b7a08d04SJames Feist                         BMCWEB_LOG_ERROR << "Unexpected configuration";
285b7a08d04SJames Feist                         messages::internalError(asyncResp->res);
286b7a08d04SJames Feist                         return;
287b7a08d04SJames Feist                     }
288b7a08d04SJames Feist 
289b7a08d04SJames Feist                     // used for making maps out of 2 vectors
290b7a08d04SJames Feist                     const std::vector<double>* keys = nullptr;
291b7a08d04SJames Feist                     const std::vector<double>* values = nullptr;
292b7a08d04SJames Feist 
293b7a08d04SJames Feist                     for (const auto& propertyPair : intfPair.second)
294b7a08d04SJames Feist                     {
295b7a08d04SJames Feist                         if (propertyPair.first == "Type" ||
296b7a08d04SJames Feist                             propertyPair.first == "Class" ||
297b7a08d04SJames Feist                             propertyPair.first == "Name")
298b7a08d04SJames Feist                         {
299b7a08d04SJames Feist                             continue;
300b7a08d04SJames Feist                         }
301b7a08d04SJames Feist 
302b7a08d04SJames Feist                         // zones
303b7a08d04SJames Feist                         if (intfPair.first == pidZoneConfigurationIface)
304b7a08d04SJames Feist                         {
305b7a08d04SJames Feist                             const double* ptr =
306*abf2add6SEd Tanous                                 std::get_if<double>(&propertyPair.second);
307b7a08d04SJames Feist                             if (ptr == nullptr)
308b7a08d04SJames Feist                             {
309b7a08d04SJames Feist                                 BMCWEB_LOG_ERROR << "Field Illegal "
310b7a08d04SJames Feist                                                  << propertyPair.first;
311b7a08d04SJames Feist                                 messages::internalError(asyncResp->res);
312b7a08d04SJames Feist                                 return;
313b7a08d04SJames Feist                             }
314b7a08d04SJames Feist                             (*config)[propertyPair.first] = *ptr;
315b7a08d04SJames Feist                         }
316b7a08d04SJames Feist 
317b7a08d04SJames Feist                         if (intfPair.first == stepwiseConfigurationIface)
318b7a08d04SJames Feist                         {
319b7a08d04SJames Feist                             if (propertyPair.first == "Reading" ||
320b7a08d04SJames Feist                                 propertyPair.first == "Output")
321b7a08d04SJames Feist                             {
322b7a08d04SJames Feist                                 const std::vector<double>* ptr =
323*abf2add6SEd Tanous                                     std::get_if<std::vector<double>>(
324b7a08d04SJames Feist                                         &propertyPair.second);
325b7a08d04SJames Feist 
326b7a08d04SJames Feist                                 if (ptr == nullptr)
327b7a08d04SJames Feist                                 {
328b7a08d04SJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
329b7a08d04SJames Feist                                                      << propertyPair.first;
330b7a08d04SJames Feist                                     messages::internalError(asyncResp->res);
331b7a08d04SJames Feist                                     return;
332b7a08d04SJames Feist                                 }
333b7a08d04SJames Feist 
334b7a08d04SJames Feist                                 if (propertyPair.first == "Reading")
335b7a08d04SJames Feist                                 {
336b7a08d04SJames Feist                                     keys = ptr;
337b7a08d04SJames Feist                                 }
338b7a08d04SJames Feist                                 else
339b7a08d04SJames Feist                                 {
340b7a08d04SJames Feist                                     values = ptr;
341b7a08d04SJames Feist                                 }
342b7a08d04SJames Feist                                 if (keys && values)
343b7a08d04SJames Feist                                 {
344b7a08d04SJames Feist                                     if (keys->size() != values->size())
345b7a08d04SJames Feist                                     {
346b7a08d04SJames Feist                                         BMCWEB_LOG_ERROR
347b7a08d04SJames Feist                                             << "Reading and Output size don't "
348b7a08d04SJames Feist                                                "match ";
349b7a08d04SJames Feist                                         messages::internalError(asyncResp->res);
350b7a08d04SJames Feist                                         return;
351b7a08d04SJames Feist                                     }
352b7a08d04SJames Feist                                     nlohmann::json& steps = (*config)["Steps"];
353b7a08d04SJames Feist                                     steps = nlohmann::json::array();
354b7a08d04SJames Feist                                     for (size_t ii = 0; ii < keys->size(); ii++)
355b7a08d04SJames Feist                                     {
356b7a08d04SJames Feist                                         steps.push_back(
357b7a08d04SJames Feist                                             {{"Target", (*keys)[ii]},
358b7a08d04SJames Feist                                              {"Output", (*values)[ii]}});
359b7a08d04SJames Feist                                     }
360b7a08d04SJames Feist                                 }
361b7a08d04SJames Feist                             }
362b7a08d04SJames Feist                             if (propertyPair.first == "NegativeHysteresis" ||
363b7a08d04SJames Feist                                 propertyPair.first == "PositiveHysteresis")
364b7a08d04SJames Feist                             {
365b7a08d04SJames Feist                                 const double* ptr =
366*abf2add6SEd Tanous                                     std::get_if<double>(&propertyPair.second);
367b7a08d04SJames Feist                                 if (ptr == nullptr)
368b7a08d04SJames Feist                                 {
369b7a08d04SJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
370b7a08d04SJames Feist                                                      << propertyPair.first;
371b7a08d04SJames Feist                                     messages::internalError(asyncResp->res);
372b7a08d04SJames Feist                                     return;
373b7a08d04SJames Feist                                 }
374b7a08d04SJames Feist                                 (*config)[propertyPair.first] = *ptr;
375b7a08d04SJames Feist                             }
376b7a08d04SJames Feist                         }
377b7a08d04SJames Feist 
378b7a08d04SJames Feist                         // pid and fans are off the same configuration
379b7a08d04SJames Feist                         if (intfPair.first == pidConfigurationIface ||
380b7a08d04SJames Feist                             intfPair.first == stepwiseConfigurationIface)
381b7a08d04SJames Feist                         {
3825b4aa86bSJames Feist 
3835b4aa86bSJames Feist                             if (propertyPair.first == "Zones")
3845b4aa86bSJames Feist                             {
3855b4aa86bSJames Feist                                 const std::vector<std::string>* inputs =
386*abf2add6SEd Tanous                                     std::get_if<std::vector<std::string>>(
3871b6b96c5SEd Tanous                                         &propertyPair.second);
3885b4aa86bSJames Feist 
3895b4aa86bSJames Feist                                 if (inputs == nullptr)
3905b4aa86bSJames Feist                                 {
3915b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR
3925b4aa86bSJames Feist                                         << "Zones Pid Field Illegal";
393a08b46ccSJason M. Bills                                     messages::internalError(asyncResp->res);
3945b4aa86bSJames Feist                                     return;
3955b4aa86bSJames Feist                                 }
396b7a08d04SJames Feist                                 auto& data = (*config)[propertyPair.first];
3975b4aa86bSJames Feist                                 data = nlohmann::json::array();
3985b4aa86bSJames Feist                                 for (std::string itemCopy : *inputs)
3995b4aa86bSJames Feist                                 {
4005b4aa86bSJames Feist                                     dbus::utility::escapePathForDbus(itemCopy);
4015b4aa86bSJames Feist                                     data.push_back(
4025b4aa86bSJames Feist                                         {{"@odata.id",
4035b4aa86bSJames Feist                                           "/redfish/v1/Managers/bmc#/Oem/"
4045b4aa86bSJames Feist                                           "OpenBmc/Fan/FanZones/" +
4055b4aa86bSJames Feist                                               itemCopy}});
4065b4aa86bSJames Feist                                 }
4075b4aa86bSJames Feist                             }
4085b4aa86bSJames Feist                             // todo(james): may never happen, but this
4095b4aa86bSJames Feist                             // assumes configuration data referenced in the
4105b4aa86bSJames Feist                             // PID config is provided by the same daemon, we
4115b4aa86bSJames Feist                             // could add another loop to cover all cases,
4125b4aa86bSJames Feist                             // but I'm okay kicking this can down the road a
4135b4aa86bSJames Feist                             // bit
4145b4aa86bSJames Feist 
4155b4aa86bSJames Feist                             else if (propertyPair.first == "Inputs" ||
4165b4aa86bSJames Feist                                      propertyPair.first == "Outputs")
4175b4aa86bSJames Feist                             {
418b7a08d04SJames Feist                                 auto& data = (*config)[propertyPair.first];
4195b4aa86bSJames Feist                                 const std::vector<std::string>* inputs =
420*abf2add6SEd Tanous                                     std::get_if<std::vector<std::string>>(
4211b6b96c5SEd Tanous                                         &propertyPair.second);
4225b4aa86bSJames Feist 
4235b4aa86bSJames Feist                                 if (inputs == nullptr)
4245b4aa86bSJames Feist                                 {
4255b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
4265b4aa86bSJames Feist                                                      << propertyPair.first;
427f12894f8SJason M. Bills                                     messages::internalError(asyncResp->res);
4285b4aa86bSJames Feist                                     return;
4295b4aa86bSJames Feist                                 }
4305b4aa86bSJames Feist                                 data = *inputs;
4315b4aa86bSJames Feist                             } // doubles
4325b4aa86bSJames Feist                             else if (propertyPair.first ==
4335b4aa86bSJames Feist                                          "FFGainCoefficient" ||
4345b4aa86bSJames Feist                                      propertyPair.first == "FFOffCoefficient" ||
4355b4aa86bSJames Feist                                      propertyPair.first == "ICoefficient" ||
4365b4aa86bSJames Feist                                      propertyPair.first == "ILimitMax" ||
4375b4aa86bSJames Feist                                      propertyPair.first == "ILimitMin" ||
4385b4aa86bSJames Feist                                      propertyPair.first == "OutLimitMax" ||
4395b4aa86bSJames Feist                                      propertyPair.first == "OutLimitMin" ||
4405b4aa86bSJames Feist                                      propertyPair.first == "PCoefficient" ||
4417625cb81SJames Feist                                      propertyPair.first == "SetPoint" ||
4425b4aa86bSJames Feist                                      propertyPair.first == "SlewNeg" ||
4435b4aa86bSJames Feist                                      propertyPair.first == "SlewPos")
4445b4aa86bSJames Feist                             {
4455b4aa86bSJames Feist                                 const double* ptr =
446*abf2add6SEd Tanous                                     std::get_if<double>(&propertyPair.second);
4475b4aa86bSJames Feist                                 if (ptr == nullptr)
4485b4aa86bSJames Feist                                 {
4495b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
4505b4aa86bSJames Feist                                                      << propertyPair.first;
451f12894f8SJason M. Bills                                     messages::internalError(asyncResp->res);
4525b4aa86bSJames Feist                                     return;
4535b4aa86bSJames Feist                                 }
454b7a08d04SJames Feist                                 (*config)[propertyPair.first] = *ptr;
4555b4aa86bSJames Feist                             }
4565b4aa86bSJames Feist                         }
4575b4aa86bSJames Feist                     }
4585b4aa86bSJames Feist                 }
4595b4aa86bSJames Feist             }
4605b4aa86bSJames Feist         },
4615b4aa86bSJames Feist         connection, path, objectManagerIface, "GetManagedObjects");
4625b4aa86bSJames Feist }
463ca537928SJennifer Lee 
46483ff9ab6SJames Feist enum class CreatePIDRet
46583ff9ab6SJames Feist {
46683ff9ab6SJames Feist     fail,
46783ff9ab6SJames Feist     del,
46883ff9ab6SJames Feist     patch
46983ff9ab6SJames Feist };
47083ff9ab6SJames Feist 
4715f2caaefSJames Feist static bool getZonesFromJsonReq(const std::shared_ptr<AsyncResp>& response,
4725f2caaefSJames Feist                                 std::vector<nlohmann::json>& config,
4735f2caaefSJames Feist                                 std::vector<std::string>& zones)
4745f2caaefSJames Feist {
4755f2caaefSJames Feist 
4765f2caaefSJames Feist     for (auto& odata : config)
4775f2caaefSJames Feist     {
4785f2caaefSJames Feist         std::string path;
4795f2caaefSJames Feist         if (!redfish::json_util::readJson(odata, response->res, "@odata.id",
4805f2caaefSJames Feist                                           path))
4815f2caaefSJames Feist         {
4825f2caaefSJames Feist             return false;
4835f2caaefSJames Feist         }
4845f2caaefSJames Feist         std::string input;
4855f2caaefSJames Feist         if (!dbus::utility::getNthStringFromPath(path, 4, input))
4865f2caaefSJames Feist         {
4875f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Got invalid path " << path;
4885f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Illegal Type Zones";
4895f2caaefSJames Feist             messages::propertyValueFormatError(response->res, odata.dump(),
4905f2caaefSJames Feist                                                "Zones");
4915f2caaefSJames Feist             return false;
4925f2caaefSJames Feist         }
4935f2caaefSJames Feist         boost::replace_all(input, "_", " ");
4945f2caaefSJames Feist         zones.emplace_back(std::move(input));
4955f2caaefSJames Feist     }
4965f2caaefSJames Feist     return true;
4975f2caaefSJames Feist }
4985f2caaefSJames Feist 
49983ff9ab6SJames Feist static CreatePIDRet createPidInterface(
50083ff9ab6SJames Feist     const std::shared_ptr<AsyncResp>& response, const std::string& type,
5015f2caaefSJames Feist     nlohmann::json&& record, const std::string& path,
50283ff9ab6SJames Feist     const dbus::utility::ManagedObjectType& managedObj, bool createNewObject,
50383ff9ab6SJames Feist     boost::container::flat_map<std::string, dbus::utility::DbusVariantType>&
50483ff9ab6SJames Feist         output,
50583ff9ab6SJames Feist     std::string& chassis)
50683ff9ab6SJames Feist {
50783ff9ab6SJames Feist 
5085f2caaefSJames Feist     // common deleter
5095f2caaefSJames Feist     if (record == nullptr)
5105f2caaefSJames Feist     {
5115f2caaefSJames Feist         std::string iface;
5125f2caaefSJames Feist         if (type == "PidControllers" || type == "FanControllers")
5135f2caaefSJames Feist         {
5145f2caaefSJames Feist             iface = pidConfigurationIface;
5155f2caaefSJames Feist         }
5165f2caaefSJames Feist         else if (type == "FanZones")
5175f2caaefSJames Feist         {
5185f2caaefSJames Feist             iface = pidZoneConfigurationIface;
5195f2caaefSJames Feist         }
5205f2caaefSJames Feist         else if (type == "StepwiseControllers")
5215f2caaefSJames Feist         {
5225f2caaefSJames Feist             iface = stepwiseConfigurationIface;
5235f2caaefSJames Feist         }
5245f2caaefSJames Feist         else
5255f2caaefSJames Feist         {
5265f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Type "
5275f2caaefSJames Feist                              << type;
5285f2caaefSJames Feist             messages::propertyUnknown(response->res, type);
5295f2caaefSJames Feist             return CreatePIDRet::fail;
5305f2caaefSJames Feist         }
5315f2caaefSJames Feist         // delete interface
5325f2caaefSJames Feist         crow::connections::systemBus->async_method_call(
5335f2caaefSJames Feist             [response, path](const boost::system::error_code ec) {
5345f2caaefSJames Feist                 if (ec)
5355f2caaefSJames Feist                 {
5365f2caaefSJames Feist                     BMCWEB_LOG_ERROR << "Error patching " << path << ": " << ec;
5375f2caaefSJames Feist                     messages::internalError(response->res);
5385f2caaefSJames Feist                 }
5395f2caaefSJames Feist             },
5405f2caaefSJames Feist             "xyz.openbmc_project.EntityManager", path, iface, "Delete");
5415f2caaefSJames Feist         return CreatePIDRet::del;
5425f2caaefSJames Feist     }
5435f2caaefSJames Feist 
54483ff9ab6SJames Feist     if (type == "PidControllers" || type == "FanControllers")
54583ff9ab6SJames Feist     {
54683ff9ab6SJames Feist         if (createNewObject)
54783ff9ab6SJames Feist         {
54883ff9ab6SJames Feist             output["Class"] = type == "PidControllers" ? std::string("temp")
54983ff9ab6SJames Feist                                                        : std::string("fan");
55083ff9ab6SJames Feist             output["Type"] = std::string("Pid");
55183ff9ab6SJames Feist         }
5525f2caaefSJames Feist 
5535f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> zones;
5545f2caaefSJames Feist         std::optional<std::vector<std::string>> inputs;
5555f2caaefSJames Feist         std::optional<std::vector<std::string>> outputs;
5565f2caaefSJames Feist         std::map<std::string, std::optional<double>> doubles;
5575f2caaefSJames Feist         if (!redfish::json_util::readJson(
5585f2caaefSJames Feist                 record, response->res, "Inputs", inputs, "Outputs", outputs,
5595f2caaefSJames Feist                 "Zones", zones, "FFGainCoefficient",
5605f2caaefSJames Feist                 doubles["FFGainCoefficient"], "FFOffCoefficient",
5615f2caaefSJames Feist                 doubles["FFOffCoefficient"], "ICoefficient",
5625f2caaefSJames Feist                 doubles["ICoefficient"], "ILimitMax", doubles["ILimitMax"],
5635f2caaefSJames Feist                 "ILimitMin", doubles["ILimitMin"], "OutLimitMax",
5645f2caaefSJames Feist                 doubles["OutLimitMax"], "OutLimitMin", doubles["OutLimitMin"],
5655f2caaefSJames Feist                 "PCoefficient", doubles["PCoefficient"], "SetPoint",
5665f2caaefSJames Feist                 doubles["SetPoint"], "SlewNeg", doubles["SlewNeg"], "SlewPos",
5675f2caaefSJames Feist                 doubles["SlewPos"]))
56883ff9ab6SJames Feist         {
5695f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property "
5705f2caaefSJames Feist                              << record.dump();
5715f2caaefSJames Feist             return CreatePIDRet::fail;
57283ff9ab6SJames Feist         }
5735f2caaefSJames Feist         if (zones)
5745f2caaefSJames Feist         {
5755f2caaefSJames Feist             std::vector<std::string> zonesStr;
5765f2caaefSJames Feist             if (!getZonesFromJsonReq(response, *zones, zonesStr))
5775f2caaefSJames Feist             {
5785f2caaefSJames Feist                 BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Zones";
5795f2caaefSJames Feist                 return CreatePIDRet::fail;
5805f2caaefSJames Feist             }
5815f2caaefSJames Feist             output["Zones"] = std::move(zonesStr);
5825f2caaefSJames Feist         }
5835f2caaefSJames Feist         if (inputs || outputs)
5845f2caaefSJames Feist         {
5855f2caaefSJames Feist             std::array<std::optional<std::vector<std::string>>*, 2> containers =
5865f2caaefSJames Feist                 {&inputs, &outputs};
5875f2caaefSJames Feist             size_t index = 0;
5885f2caaefSJames Feist             for (const auto& containerPtr : containers)
5895f2caaefSJames Feist             {
5905f2caaefSJames Feist                 std::optional<std::vector<std::string>>& container =
5915f2caaefSJames Feist                     *containerPtr;
5925f2caaefSJames Feist                 if (!container)
5935f2caaefSJames Feist                 {
5945f2caaefSJames Feist                     index++;
5955f2caaefSJames Feist                     continue;
59683ff9ab6SJames Feist                 }
59783ff9ab6SJames Feist 
5985f2caaefSJames Feist                 for (std::string& value : *container)
59983ff9ab6SJames Feist                 {
60083ff9ab6SJames Feist 
60183ff9ab6SJames Feist                     // try to find the sensor in the
60283ff9ab6SJames Feist                     // configuration
60383ff9ab6SJames Feist                     if (chassis.empty())
60483ff9ab6SJames Feist                     {
6055f2caaefSJames Feist                         std::string escaped =
6065f2caaefSJames Feist                             boost::replace_all_copy(value, " ", "_");
60783ff9ab6SJames Feist                         std::find_if(
60883ff9ab6SJames Feist                             managedObj.begin(), managedObj.end(),
6095f2caaefSJames Feist                             [&chassis, &escaped](const auto& obj) {
61083ff9ab6SJames Feist                                 if (boost::algorithm::ends_with(obj.first.str,
6115f2caaefSJames Feist                                                                 escaped))
61283ff9ab6SJames Feist                                 {
61383ff9ab6SJames Feist                                     return dbus::utility::getNthStringFromPath(
61483ff9ab6SJames Feist                                         obj.first.str, 5, chassis);
61583ff9ab6SJames Feist                                 }
61683ff9ab6SJames Feist                                 return false;
61783ff9ab6SJames Feist                             });
61883ff9ab6SJames Feist                     }
6195f2caaefSJames Feist                     boost::replace_all(value, "_", " ");
62083ff9ab6SJames Feist                 }
6215f2caaefSJames Feist                 std::string key;
6225f2caaefSJames Feist                 if (index == 0)
6235f2caaefSJames Feist                 {
6245f2caaefSJames Feist                     key = "Inputs";
6255f2caaefSJames Feist                 }
6265f2caaefSJames Feist                 else
6275f2caaefSJames Feist                 {
6285f2caaefSJames Feist                     key = "Outputs";
6295f2caaefSJames Feist                 }
6305f2caaefSJames Feist                 output[key] = *container;
6315f2caaefSJames Feist                 index++;
6325f2caaefSJames Feist             }
63383ff9ab6SJames Feist         }
63483ff9ab6SJames Feist 
63583ff9ab6SJames Feist         // doubles
6365f2caaefSJames Feist         for (const auto& pairs : doubles)
63783ff9ab6SJames Feist         {
6385f2caaefSJames Feist             if (!pairs.second)
63983ff9ab6SJames Feist             {
6405f2caaefSJames Feist                 continue;
64183ff9ab6SJames Feist             }
6425f2caaefSJames Feist             BMCWEB_LOG_DEBUG << pairs.first << " = " << *pairs.second;
6435f2caaefSJames Feist             output[pairs.first] = *(pairs.second);
6445f2caaefSJames Feist         }
64583ff9ab6SJames Feist     }
64683ff9ab6SJames Feist 
64783ff9ab6SJames Feist     else if (type == "FanZones")
64883ff9ab6SJames Feist     {
64983ff9ab6SJames Feist         output["Type"] = std::string("Pid.Zone");
65083ff9ab6SJames Feist 
6515f2caaefSJames Feist         std::optional<nlohmann::json> chassisContainer;
6525f2caaefSJames Feist         std::optional<double> failSafePercent;
6535f2caaefSJames Feist         std::optional<double> minThermalRpm;
6545f2caaefSJames Feist         if (!redfish::json_util::readJson(record, response->res, "Chassis",
6555f2caaefSJames Feist                                           chassisContainer, "FailSafePercent",
6565f2caaefSJames Feist                                           failSafePercent, "MinThermalRpm",
6575f2caaefSJames Feist                                           minThermalRpm))
65883ff9ab6SJames Feist         {
6595f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property "
6605f2caaefSJames Feist                              << record.dump();
66183ff9ab6SJames Feist             return CreatePIDRet::fail;
66283ff9ab6SJames Feist         }
6635f2caaefSJames Feist 
6645f2caaefSJames Feist         if (chassisContainer)
66583ff9ab6SJames Feist         {
6665f2caaefSJames Feist 
6675f2caaefSJames Feist             std::string chassisId;
6685f2caaefSJames Feist             if (!redfish::json_util::readJson(*chassisContainer, response->res,
6695f2caaefSJames Feist                                               "@odata.id", chassisId))
6705f2caaefSJames Feist             {
6715f2caaefSJames Feist                 BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property "
6725f2caaefSJames Feist                                  << chassisContainer->dump();
67383ff9ab6SJames Feist                 return CreatePIDRet::fail;
67483ff9ab6SJames Feist             }
67583ff9ab6SJames Feist 
67683ff9ab6SJames Feist             // /refish/v1/chassis/chassis_name/
6775f2caaefSJames Feist             if (!dbus::utility::getNthStringFromPath(chassisId, 3, chassis))
67883ff9ab6SJames Feist             {
6795f2caaefSJames Feist                 BMCWEB_LOG_ERROR << "Got invalid path " << chassisId;
6805f2caaefSJames Feist                 messages::invalidObject(response->res, chassisId);
68183ff9ab6SJames Feist                 return CreatePIDRet::fail;
68283ff9ab6SJames Feist             }
68383ff9ab6SJames Feist         }
6845f2caaefSJames Feist         if (minThermalRpm)
68583ff9ab6SJames Feist         {
6865f2caaefSJames Feist             output["MinThermalRpm"] = *minThermalRpm;
6875f2caaefSJames Feist         }
6885f2caaefSJames Feist         if (failSafePercent)
68983ff9ab6SJames Feist         {
6905f2caaefSJames Feist             output["FailSafePercent"] = *failSafePercent;
6915f2caaefSJames Feist         }
6925f2caaefSJames Feist     }
6935f2caaefSJames Feist     else if (type == "StepwiseControllers")
6945f2caaefSJames Feist     {
6955f2caaefSJames Feist         output["Type"] = std::string("Stepwise");
6965f2caaefSJames Feist 
6975f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> zones;
6985f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> steps;
6995f2caaefSJames Feist         std::optional<std::vector<std::string>> inputs;
7005f2caaefSJames Feist         std::optional<double> positiveHysteresis;
7015f2caaefSJames Feist         std::optional<double> negativeHysteresis;
7025f2caaefSJames Feist         if (!redfish::json_util::readJson(
7035f2caaefSJames Feist                 record, response->res, "Zones", zones, "Steps", steps, "Inputs",
7045f2caaefSJames Feist                 inputs, "PositiveHysteresis", positiveHysteresis,
7055f2caaefSJames Feist                 "NegativeHysteresis", negativeHysteresis))
7065f2caaefSJames Feist         {
7075f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property "
7085f2caaefSJames Feist                              << record.dump();
70983ff9ab6SJames Feist             return CreatePIDRet::fail;
71083ff9ab6SJames Feist         }
7115f2caaefSJames Feist 
7125f2caaefSJames Feist         if (zones)
71383ff9ab6SJames Feist         {
7145f2caaefSJames Feist             std::vector<std::string> zoneStrs;
7155f2caaefSJames Feist             if (!getZonesFromJsonReq(response, *zones, zoneStrs))
7165f2caaefSJames Feist             {
7175f2caaefSJames Feist                 BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Zones";
71883ff9ab6SJames Feist                 return CreatePIDRet::fail;
71983ff9ab6SJames Feist             }
7205f2caaefSJames Feist             output["Zones"] = std::move(zoneStrs);
7215f2caaefSJames Feist         }
7225f2caaefSJames Feist         if (steps)
7235f2caaefSJames Feist         {
7245f2caaefSJames Feist             std::vector<double> readings;
7255f2caaefSJames Feist             std::vector<double> outputs;
7265f2caaefSJames Feist             for (auto& step : *steps)
7275f2caaefSJames Feist             {
7285f2caaefSJames Feist                 double target;
7295f2caaefSJames Feist                 double output;
7305f2caaefSJames Feist 
7315f2caaefSJames Feist                 if (!redfish::json_util::readJson(step, response->res, "Target",
7325f2caaefSJames Feist                                                   target, "Output", output))
7335f2caaefSJames Feist                 {
7345f2caaefSJames Feist                     BMCWEB_LOG_ERROR << "Line:" << __LINE__
7355f2caaefSJames Feist                                      << ", Illegal Property " << record.dump();
7365f2caaefSJames Feist                     return CreatePIDRet::fail;
7375f2caaefSJames Feist                 }
7385f2caaefSJames Feist                 readings.emplace_back(target);
7395f2caaefSJames Feist                 outputs.emplace_back(output);
7405f2caaefSJames Feist             }
7415f2caaefSJames Feist             output["Reading"] = std::move(readings);
7425f2caaefSJames Feist             output["Output"] = std::move(outputs);
7435f2caaefSJames Feist         }
7445f2caaefSJames Feist         if (inputs)
7455f2caaefSJames Feist         {
7465f2caaefSJames Feist             for (std::string& value : *inputs)
7475f2caaefSJames Feist             {
7485f2caaefSJames Feist                 if (chassis.empty())
7495f2caaefSJames Feist                 {
7505f2caaefSJames Feist                     std::string escaped =
7515f2caaefSJames Feist                         boost::replace_all_copy(value, " ", "_");
7525f2caaefSJames Feist                     std::find_if(
7535f2caaefSJames Feist                         managedObj.begin(), managedObj.end(),
7545f2caaefSJames Feist                         [&chassis, &escaped](const auto& obj) {
7555f2caaefSJames Feist                             if (boost::algorithm::ends_with(obj.first.str,
7565f2caaefSJames Feist                                                             escaped))
7575f2caaefSJames Feist                             {
7585f2caaefSJames Feist                                 return dbus::utility::getNthStringFromPath(
7595f2caaefSJames Feist                                     obj.first.str, 5, chassis);
7605f2caaefSJames Feist                             }
7615f2caaefSJames Feist                             return false;
7625f2caaefSJames Feist                         });
7635f2caaefSJames Feist                 }
7645f2caaefSJames Feist                 boost::replace_all(value, "_", " ");
7655f2caaefSJames Feist             }
7665f2caaefSJames Feist             output["Inputs"] = std::move(*inputs);
7675f2caaefSJames Feist         }
7685f2caaefSJames Feist         if (negativeHysteresis)
7695f2caaefSJames Feist         {
7705f2caaefSJames Feist             output["NegativeHysteresis"] = *negativeHysteresis;
7715f2caaefSJames Feist         }
7725f2caaefSJames Feist         if (positiveHysteresis)
7735f2caaefSJames Feist         {
7745f2caaefSJames Feist             output["PositiveHysteresis"] = *positiveHysteresis;
77583ff9ab6SJames Feist         }
77683ff9ab6SJames Feist     }
77783ff9ab6SJames Feist     else
77883ff9ab6SJames Feist     {
7795f2caaefSJames Feist         BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Type " << type;
78035a62c7cSJason M. Bills         messages::propertyUnknown(response->res, type);
78183ff9ab6SJames Feist         return CreatePIDRet::fail;
78283ff9ab6SJames Feist     }
78383ff9ab6SJames Feist     return CreatePIDRet::patch;
78483ff9ab6SJames Feist }
78583ff9ab6SJames Feist 
7861abe55efSEd Tanous class Manager : public Node
7871abe55efSEd Tanous {
7889c310685SBorawski.Lukasz   public:
7895b4aa86bSJames Feist     Manager(CrowApp& app) : Node(app, "/redfish/v1/Managers/bmc/")
7901abe55efSEd Tanous     {
7910f74e643SEd Tanous         uuid = app.template getMiddleware<crow::persistent_data::Middleware>()
79255c7b7a2SEd Tanous                    .systemUuid;
793a434f2bdSEd Tanous         entityPrivileges = {
794a434f2bdSEd Tanous             {boost::beast::http::verb::get, {{"Login"}}},
795e0d918bcSEd Tanous             {boost::beast::http::verb::head, {{"Login"}}},
796e0d918bcSEd Tanous             {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
797e0d918bcSEd Tanous             {boost::beast::http::verb::put, {{"ConfigureManager"}}},
798e0d918bcSEd Tanous             {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
799e0d918bcSEd Tanous             {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
8009c310685SBorawski.Lukasz     }
8019c310685SBorawski.Lukasz 
8029c310685SBorawski.Lukasz   private:
8035b4aa86bSJames Feist     void getPidValues(std::shared_ptr<AsyncResp> asyncResp)
8045b4aa86bSJames Feist     {
8055b4aa86bSJames Feist         crow::connections::systemBus->async_method_call(
8065b4aa86bSJames Feist             [asyncResp](const boost::system::error_code ec,
8075b4aa86bSJames Feist                         const crow::openbmc_mapper::GetSubTreeType& subtree) {
8085b4aa86bSJames Feist                 if (ec)
8095b4aa86bSJames Feist                 {
8105b4aa86bSJames Feist                     BMCWEB_LOG_ERROR << ec;
811f12894f8SJason M. Bills                     messages::internalError(asyncResp->res);
8125b4aa86bSJames Feist                     return;
8135b4aa86bSJames Feist                 }
8145b4aa86bSJames Feist 
8155b4aa86bSJames Feist                 // create map of <connection, path to objMgr>>
8165b4aa86bSJames Feist                 boost::container::flat_map<std::string, std::string>
8175b4aa86bSJames Feist                     objectMgrPaths;
8186bce33bcSJames Feist                 boost::container::flat_set<std::string> calledConnections;
8195b4aa86bSJames Feist                 for (const auto& pathGroup : subtree)
8205b4aa86bSJames Feist                 {
8215b4aa86bSJames Feist                     for (const auto& connectionGroup : pathGroup.second)
8225b4aa86bSJames Feist                     {
8236bce33bcSJames Feist                         auto findConnection =
8246bce33bcSJames Feist                             calledConnections.find(connectionGroup.first);
8256bce33bcSJames Feist                         if (findConnection != calledConnections.end())
8266bce33bcSJames Feist                         {
8276bce33bcSJames Feist                             break;
8286bce33bcSJames Feist                         }
8295b4aa86bSJames Feist                         for (const std::string& interface :
8305b4aa86bSJames Feist                              connectionGroup.second)
8315b4aa86bSJames Feist                         {
8325b4aa86bSJames Feist                             if (interface == objectManagerIface)
8335b4aa86bSJames Feist                             {
8345b4aa86bSJames Feist                                 objectMgrPaths[connectionGroup.first] =
8355b4aa86bSJames Feist                                     pathGroup.first;
8365b4aa86bSJames Feist                             }
8375b4aa86bSJames Feist                             // this list is alphabetical, so we
8385b4aa86bSJames Feist                             // should have found the objMgr by now
8395b4aa86bSJames Feist                             if (interface == pidConfigurationIface ||
840b7a08d04SJames Feist                                 interface == pidZoneConfigurationIface ||
841b7a08d04SJames Feist                                 interface == stepwiseConfigurationIface)
8425b4aa86bSJames Feist                             {
8435b4aa86bSJames Feist                                 auto findObjMgr =
8445b4aa86bSJames Feist                                     objectMgrPaths.find(connectionGroup.first);
8455b4aa86bSJames Feist                                 if (findObjMgr == objectMgrPaths.end())
8465b4aa86bSJames Feist                                 {
8475b4aa86bSJames Feist                                     BMCWEB_LOG_DEBUG << connectionGroup.first
8485b4aa86bSJames Feist                                                      << "Has no Object Manager";
8495b4aa86bSJames Feist                                     continue;
8505b4aa86bSJames Feist                                 }
8516bce33bcSJames Feist 
8526bce33bcSJames Feist                                 calledConnections.insert(connectionGroup.first);
8536bce33bcSJames Feist 
8545b4aa86bSJames Feist                                 asyncPopulatePid(findObjMgr->first,
8555b4aa86bSJames Feist                                                  findObjMgr->second, asyncResp);
8565b4aa86bSJames Feist                                 break;
8575b4aa86bSJames Feist                             }
8585b4aa86bSJames Feist                         }
8595b4aa86bSJames Feist                     }
8605b4aa86bSJames Feist                 }
8615b4aa86bSJames Feist             },
8625b4aa86bSJames Feist             "xyz.openbmc_project.ObjectMapper",
8635b4aa86bSJames Feist             "/xyz/openbmc_project/object_mapper",
8645b4aa86bSJames Feist             "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0,
865b7a08d04SJames Feist             std::array<const char*, 4>{
866b7a08d04SJames Feist                 pidConfigurationIface, pidZoneConfigurationIface,
867b7a08d04SJames Feist                 objectManagerIface, stepwiseConfigurationIface});
8685b4aa86bSJames Feist     }
8695b4aa86bSJames Feist 
87055c7b7a2SEd Tanous     void doGet(crow::Response& res, const crow::Request& req,
8711abe55efSEd Tanous                const std::vector<std::string>& params) override
8721abe55efSEd Tanous     {
8730f74e643SEd Tanous         res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc";
8740f74e643SEd Tanous         res.jsonValue["@odata.type"] = "#Manager.v1_3_0.Manager";
8750f74e643SEd Tanous         res.jsonValue["@odata.context"] =
8760f74e643SEd Tanous             "/redfish/v1/$metadata#Manager.Manager";
8770f74e643SEd Tanous         res.jsonValue["Id"] = "bmc";
8780f74e643SEd Tanous         res.jsonValue["Name"] = "OpenBmc Manager";
8790f74e643SEd Tanous         res.jsonValue["Description"] = "Baseboard Management Controller";
8800f74e643SEd Tanous         res.jsonValue["PowerState"] = "On";
8810f74e643SEd Tanous         res.jsonValue["ManagerType"] = "BMC";
8827517658eSEd Tanous         res.jsonValue["UUID"] = uuid;
8830f74e643SEd Tanous         res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model
8840f74e643SEd Tanous 
8850f74e643SEd Tanous         res.jsonValue["LogServices"] = {
8860f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/LogServices"}};
8870f74e643SEd Tanous 
8880f74e643SEd Tanous         res.jsonValue["NetworkProtocol"] = {
8890f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/NetworkProtocol"}};
8900f74e643SEd Tanous 
8910f74e643SEd Tanous         res.jsonValue["EthernetInterfaces"] = {
8920f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/EthernetInterfaces"}};
8930f74e643SEd Tanous         // default oem data
8940f74e643SEd Tanous         nlohmann::json& oem = res.jsonValue["Oem"];
8950f74e643SEd Tanous         nlohmann::json& oemOpenbmc = oem["OpenBmc"];
8960f74e643SEd Tanous         oem["@odata.type"] = "#OemManager.Oem";
8970f74e643SEd Tanous         oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem";
8980f74e643SEd Tanous         oem["@odata.context"] = "/redfish/v1/$metadata#OemManager.Oem";
8990f74e643SEd Tanous         oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc";
9000f74e643SEd Tanous         oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc";
9010f74e643SEd Tanous         oemOpenbmc["@odata.context"] =
9020f74e643SEd Tanous             "/redfish/v1/$metadata#OemManager.OpenBmc";
9030f74e643SEd Tanous 
904ed5befbdSJennifer Lee         // Update Actions object.
9050f74e643SEd Tanous         nlohmann::json& manager_reset =
9060f74e643SEd Tanous             res.jsonValue["Actions"]["#Manager.Reset"];
907ed5befbdSJennifer Lee         manager_reset["target"] =
908ed5befbdSJennifer Lee             "/redfish/v1/Managers/bmc/Actions/Manager.Reset";
909ed5befbdSJennifer Lee         manager_reset["ResetType@Redfish.AllowableValues"] = {
910ed5befbdSJennifer Lee             "GracefulRestart"};
911ca537928SJennifer Lee 
9120f74e643SEd Tanous         res.jsonValue["DateTime"] = getDateTime();
913ed5befbdSJennifer Lee         std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res);
9145b4aa86bSJames Feist 
915ca537928SJennifer Lee         crow::connections::systemBus->async_method_call(
916ca537928SJennifer Lee             [asyncResp](const boost::system::error_code ec,
9175b4aa86bSJames Feist                         const dbus::utility::ManagedObjectType& resp) {
918ca537928SJennifer Lee                 if (ec)
919ca537928SJennifer Lee                 {
920ca537928SJennifer Lee                     BMCWEB_LOG_ERROR << "Error while getting Software Version";
921f12894f8SJason M. Bills                     messages::internalError(asyncResp->res);
922ca537928SJennifer Lee                     return;
923ca537928SJennifer Lee                 }
924ca537928SJennifer Lee 
925ca537928SJennifer Lee                 for (auto& objpath : resp)
926ca537928SJennifer Lee                 {
927ca537928SJennifer Lee                     for (auto& interface : objpath.second)
928ca537928SJennifer Lee                     {
9295f2caaefSJames Feist                         // If interface is
9305f2caaefSJames Feist                         // xyz.openbmc_project.Software.Version, this is
9315f2caaefSJames Feist                         // what we're looking for.
932ca537928SJennifer Lee                         if (interface.first ==
933ca537928SJennifer Lee                             "xyz.openbmc_project.Software.Version")
934ca537928SJennifer Lee                         {
935ca537928SJennifer Lee                             // Cut out everyting until last "/", ...
936ca537928SJennifer Lee                             const std::string& iface_id = objpath.first;
937ca537928SJennifer Lee                             for (auto& property : interface.second)
938ca537928SJennifer Lee                             {
939ca537928SJennifer Lee                                 if (property.first == "Version")
940ca537928SJennifer Lee                                 {
941ca537928SJennifer Lee                                     const std::string* value =
942*abf2add6SEd Tanous                                         std::get_if<std::string>(
943*abf2add6SEd Tanous                                             &property.second);
944ca537928SJennifer Lee                                     if (value == nullptr)
945ca537928SJennifer Lee                                     {
946ca537928SJennifer Lee                                         continue;
947ca537928SJennifer Lee                                     }
948ca537928SJennifer Lee                                     asyncResp->res
949ca537928SJennifer Lee                                         .jsonValue["FirmwareVersion"] = *value;
950ca537928SJennifer Lee                                 }
951ca537928SJennifer Lee                             }
952ca537928SJennifer Lee                         }
953ca537928SJennifer Lee                     }
954ca537928SJennifer Lee                 }
955ca537928SJennifer Lee             },
956ca537928SJennifer Lee             "xyz.openbmc_project.Software.BMC.Updater",
957ca537928SJennifer Lee             "/xyz/openbmc_project/software",
958ca537928SJennifer Lee             "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
9595b4aa86bSJames Feist         getPidValues(asyncResp);
9605b4aa86bSJames Feist     }
9615f2caaefSJames Feist     void setPidValues(std::shared_ptr<AsyncResp> response, nlohmann::json& data)
96283ff9ab6SJames Feist     {
9635f2caaefSJames Feist 
96483ff9ab6SJames Feist         // todo(james): might make sense to do a mapper call here if this
96583ff9ab6SJames Feist         // interface gets more traction
96683ff9ab6SJames Feist         crow::connections::systemBus->async_method_call(
96783ff9ab6SJames Feist             [response,
96883ff9ab6SJames Feist              data](const boost::system::error_code ec,
96983ff9ab6SJames Feist                    const dbus::utility::ManagedObjectType& managedObj) {
97083ff9ab6SJames Feist                 if (ec)
97183ff9ab6SJames Feist                 {
97283ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Error communicating to Entity Manager";
97335a62c7cSJason M. Bills                     messages::internalError(response->res);
97483ff9ab6SJames Feist                     return;
97583ff9ab6SJames Feist                 }
9765f2caaefSJames Feist 
9775f2caaefSJames Feist                 // todo(james) mutable doesn't work with asio bindings
9785f2caaefSJames Feist                 nlohmann::json jsonData = data;
9795f2caaefSJames Feist 
9805f2caaefSJames Feist                 std::optional<nlohmann::json> pidControllers;
9815f2caaefSJames Feist                 std::optional<nlohmann::json> fanControllers;
9825f2caaefSJames Feist                 std::optional<nlohmann::json> fanZones;
9835f2caaefSJames Feist                 std::optional<nlohmann::json> stepwiseControllers;
9845f2caaefSJames Feist                 if (!redfish::json_util::readJson(
9855f2caaefSJames Feist                         jsonData, response->res, "PidControllers",
9865f2caaefSJames Feist                         pidControllers, "FanControllers", fanControllers,
9875f2caaefSJames Feist                         "FanZones", fanZones, "StepwiseControllers",
9885f2caaefSJames Feist                         stepwiseControllers))
98983ff9ab6SJames Feist                 {
9905f2caaefSJames Feist                     BMCWEB_LOG_ERROR << "Line:" << __LINE__
9915f2caaefSJames Feist                                      << ", Illegal Property "
9925f2caaefSJames Feist                                      << jsonData.dump();
99383ff9ab6SJames Feist                     return;
99483ff9ab6SJames Feist                 }
9955f2caaefSJames Feist                 std::array<
9965f2caaefSJames Feist                     std::pair<const char*, std::optional<nlohmann::json>*>, 4>
9975f2caaefSJames Feist                     sections = {
9985f2caaefSJames Feist                         std::make_pair("PidControllers", &pidControllers),
9995f2caaefSJames Feist                         std::make_pair("FanControllers", &fanControllers),
10005f2caaefSJames Feist                         std::make_pair("FanZones", &fanZones),
10015f2caaefSJames Feist                         std::make_pair("StepwiseControllers",
10025f2caaefSJames Feist                                        &stepwiseControllers)};
10035f2caaefSJames Feist 
10045f2caaefSJames Feist                 for (auto& containerPair : sections)
100583ff9ab6SJames Feist                 {
10065f2caaefSJames Feist                     auto& container = *(containerPair.second);
10075f2caaefSJames Feist                     if (!container)
10085f2caaefSJames Feist                     {
10095f2caaefSJames Feist                         continue;
10105f2caaefSJames Feist                     }
10115f2caaefSJames Feist                     const char* type = containerPair.first;
10125f2caaefSJames Feist 
10135f2caaefSJames Feist                     for (auto& record : container->items())
10145f2caaefSJames Feist                     {
10155f2caaefSJames Feist                         const auto& name = record.key();
101683ff9ab6SJames Feist                         auto pathItr =
101783ff9ab6SJames Feist                             std::find_if(managedObj.begin(), managedObj.end(),
101883ff9ab6SJames Feist                                          [&name](const auto& obj) {
101983ff9ab6SJames Feist                                              return boost::algorithm::ends_with(
102083ff9ab6SJames Feist                                                  obj.first.str, name);
102183ff9ab6SJames Feist                                          });
102283ff9ab6SJames Feist                         boost::container::flat_map<
102383ff9ab6SJames Feist                             std::string, dbus::utility::DbusVariantType>
102483ff9ab6SJames Feist                             output;
102583ff9ab6SJames Feist 
102683ff9ab6SJames Feist                         output.reserve(16); // The pid interface length
102783ff9ab6SJames Feist 
102883ff9ab6SJames Feist                         // determines if we're patching entity-manager or
102983ff9ab6SJames Feist                         // creating a new object
103083ff9ab6SJames Feist                         bool createNewObject = (pathItr == managedObj.end());
10315f2caaefSJames Feist                         std::string iface;
10325f2caaefSJames Feist                         if (type == "PidControllers" ||
10335f2caaefSJames Feist                             type == "FanControllers")
103483ff9ab6SJames Feist                         {
10355f2caaefSJames Feist                             iface = pidConfigurationIface;
103683ff9ab6SJames Feist                             if (!createNewObject &&
103783ff9ab6SJames Feist                                 pathItr->second.find(pidConfigurationIface) ==
103883ff9ab6SJames Feist                                     pathItr->second.end())
103983ff9ab6SJames Feist                             {
104083ff9ab6SJames Feist                                 createNewObject = true;
104183ff9ab6SJames Feist                             }
104283ff9ab6SJames Feist                         }
10435f2caaefSJames Feist                         else if (type == "FanZones")
10445f2caaefSJames Feist                         {
10455f2caaefSJames Feist                             iface = pidZoneConfigurationIface;
10465f2caaefSJames Feist                             if (!createNewObject &&
104783ff9ab6SJames Feist                                 pathItr->second.find(
104883ff9ab6SJames Feist                                     pidZoneConfigurationIface) ==
104983ff9ab6SJames Feist                                     pathItr->second.end())
105083ff9ab6SJames Feist                             {
10515f2caaefSJames Feist 
105283ff9ab6SJames Feist                                 createNewObject = true;
105383ff9ab6SJames Feist                             }
10545f2caaefSJames Feist                         }
10555f2caaefSJames Feist                         else if (type == "StepwiseControllers")
10565f2caaefSJames Feist                         {
10575f2caaefSJames Feist                             iface = stepwiseConfigurationIface;
10585f2caaefSJames Feist                             if (!createNewObject &&
10595f2caaefSJames Feist                                 pathItr->second.find(
10605f2caaefSJames Feist                                     stepwiseConfigurationIface) ==
10615f2caaefSJames Feist                                     pathItr->second.end())
10625f2caaefSJames Feist                             {
10635f2caaefSJames Feist                                 createNewObject = true;
10645f2caaefSJames Feist                             }
10655f2caaefSJames Feist                         }
106683ff9ab6SJames Feist                         output["Name"] =
106783ff9ab6SJames Feist                             boost::replace_all_copy(name, "_", " ");
106883ff9ab6SJames Feist 
106983ff9ab6SJames Feist                         std::string chassis;
107083ff9ab6SJames Feist                         CreatePIDRet ret = createPidInterface(
10715f2caaefSJames Feist                             response, type, std::move(record.value()),
107283ff9ab6SJames Feist                             pathItr->first.str, managedObj, createNewObject,
107383ff9ab6SJames Feist                             output, chassis);
107483ff9ab6SJames Feist                         if (ret == CreatePIDRet::fail)
107583ff9ab6SJames Feist                         {
107683ff9ab6SJames Feist                             return;
107783ff9ab6SJames Feist                         }
107883ff9ab6SJames Feist                         else if (ret == CreatePIDRet::del)
107983ff9ab6SJames Feist                         {
108083ff9ab6SJames Feist                             continue;
108183ff9ab6SJames Feist                         }
108283ff9ab6SJames Feist 
108383ff9ab6SJames Feist                         if (!createNewObject)
108483ff9ab6SJames Feist                         {
108583ff9ab6SJames Feist                             for (const auto& property : output)
108683ff9ab6SJames Feist                             {
108783ff9ab6SJames Feist                                 crow::connections::systemBus->async_method_call(
108883ff9ab6SJames Feist                                     [response,
108983ff9ab6SJames Feist                                      propertyName{std::string(property.first)}](
109083ff9ab6SJames Feist                                         const boost::system::error_code ec) {
109183ff9ab6SJames Feist                                         if (ec)
109283ff9ab6SJames Feist                                         {
109383ff9ab6SJames Feist                                             BMCWEB_LOG_ERROR
109483ff9ab6SJames Feist                                                 << "Error patching "
109583ff9ab6SJames Feist                                                 << propertyName << ": " << ec;
109635a62c7cSJason M. Bills                                             messages::internalError(
109735a62c7cSJason M. Bills                                                 response->res);
109883ff9ab6SJames Feist                                         }
109983ff9ab6SJames Feist                                     },
110083ff9ab6SJames Feist                                     "xyz.openbmc_project.EntityManager",
110183ff9ab6SJames Feist                                     pathItr->first.str,
110283ff9ab6SJames Feist                                     "org.freedesktop.DBus.Properties", "Set",
11035f2caaefSJames Feist                                     iface, property.first, property.second);
110483ff9ab6SJames Feist                             }
110583ff9ab6SJames Feist                         }
110683ff9ab6SJames Feist                         else
110783ff9ab6SJames Feist                         {
110883ff9ab6SJames Feist                             if (chassis.empty())
110983ff9ab6SJames Feist                             {
111083ff9ab6SJames Feist                                 BMCWEB_LOG_ERROR
111183ff9ab6SJames Feist                                     << "Failed to get chassis from config";
111235a62c7cSJason M. Bills                                 messages::invalidObject(response->res, name);
111383ff9ab6SJames Feist                                 return;
111483ff9ab6SJames Feist                             }
111583ff9ab6SJames Feist 
111683ff9ab6SJames Feist                             bool foundChassis = false;
111783ff9ab6SJames Feist                             for (const auto& obj : managedObj)
111883ff9ab6SJames Feist                             {
111983ff9ab6SJames Feist                                 if (boost::algorithm::ends_with(obj.first.str,
112083ff9ab6SJames Feist                                                                 chassis))
112183ff9ab6SJames Feist                                 {
112283ff9ab6SJames Feist                                     chassis = obj.first.str;
112383ff9ab6SJames Feist                                     foundChassis = true;
112483ff9ab6SJames Feist                                     break;
112583ff9ab6SJames Feist                                 }
112683ff9ab6SJames Feist                             }
112783ff9ab6SJames Feist                             if (!foundChassis)
112883ff9ab6SJames Feist                             {
112983ff9ab6SJames Feist                                 BMCWEB_LOG_ERROR
113083ff9ab6SJames Feist                                     << "Failed to find chassis on dbus";
113183ff9ab6SJames Feist                                 messages::resourceMissingAtURI(
113235a62c7cSJason M. Bills                                     response->res,
113335a62c7cSJason M. Bills                                     "/redfish/v1/Chassis/" + chassis);
113483ff9ab6SJames Feist                                 return;
113583ff9ab6SJames Feist                             }
113683ff9ab6SJames Feist 
113783ff9ab6SJames Feist                             crow::connections::systemBus->async_method_call(
113883ff9ab6SJames Feist                                 [response](const boost::system::error_code ec) {
113983ff9ab6SJames Feist                                     if (ec)
114083ff9ab6SJames Feist                                     {
114183ff9ab6SJames Feist                                         BMCWEB_LOG_ERROR
114283ff9ab6SJames Feist                                             << "Error Adding Pid Object " << ec;
114335a62c7cSJason M. Bills                                         messages::internalError(response->res);
114483ff9ab6SJames Feist                                     }
114583ff9ab6SJames Feist                                 },
114683ff9ab6SJames Feist                                 "xyz.openbmc_project.EntityManager", chassis,
114783ff9ab6SJames Feist                                 "xyz.openbmc_project.AddObject", "AddObject",
114883ff9ab6SJames Feist                                 output);
114983ff9ab6SJames Feist                         }
115083ff9ab6SJames Feist                     }
115183ff9ab6SJames Feist                 }
115283ff9ab6SJames Feist             },
115383ff9ab6SJames Feist             "xyz.openbmc_project.EntityManager", "/", objectManagerIface,
115483ff9ab6SJames Feist             "GetManagedObjects");
115583ff9ab6SJames Feist     }
11565b4aa86bSJames Feist 
11575b4aa86bSJames Feist     void doPatch(crow::Response& res, const crow::Request& req,
11585b4aa86bSJames Feist                  const std::vector<std::string>& params) override
11595b4aa86bSJames Feist     {
11600627a2c7SEd Tanous         std::optional<nlohmann::json> oem;
11610627a2c7SEd Tanous 
11620627a2c7SEd Tanous         if (!json_util::readJson(req, res, "Oem", oem))
116383ff9ab6SJames Feist         {
116483ff9ab6SJames Feist             return;
116583ff9ab6SJames Feist         }
11660627a2c7SEd Tanous 
116783ff9ab6SJames Feist         std::shared_ptr<AsyncResp> response = std::make_shared<AsyncResp>(res);
11680627a2c7SEd Tanous 
11690627a2c7SEd Tanous         if (oem)
117083ff9ab6SJames Feist         {
11710627a2c7SEd Tanous             for (const auto& oemLevel : oem->items())
117283ff9ab6SJames Feist             {
11735f2caaefSJames Feist                 std::optional<nlohmann::json> openbmc;
11745f2caaefSJames Feist                 if (!redfish::json_util::readJson(*oem, res, "OpenBmc",
11755f2caaefSJames Feist                                                   openbmc))
117683ff9ab6SJames Feist                 {
11775f2caaefSJames Feist                     BMCWEB_LOG_ERROR << "Line:" << __LINE__
11785f2caaefSJames Feist                                      << ", Illegal Property " << oem->dump();
117983ff9ab6SJames Feist                     return;
118083ff9ab6SJames Feist                 }
11815f2caaefSJames Feist                 if (openbmc)
118283ff9ab6SJames Feist                 {
11835f2caaefSJames Feist                     std::optional<nlohmann::json> fan;
11845f2caaefSJames Feist                     if (!redfish::json_util::readJson(*openbmc, res, "Fan",
11855f2caaefSJames Feist                                                       fan))
118683ff9ab6SJames Feist                     {
11875f2caaefSJames Feist                         BMCWEB_LOG_ERROR << "Line:" << __LINE__
11885f2caaefSJames Feist                                          << ", Illegal Property "
11895f2caaefSJames Feist                                          << openbmc->dump();
119083ff9ab6SJames Feist                         return;
119183ff9ab6SJames Feist                     }
11925f2caaefSJames Feist                     if (fan)
119383ff9ab6SJames Feist                     {
11945f2caaefSJames Feist                         setPidValues(response, *fan);
119583ff9ab6SJames Feist                     }
119683ff9ab6SJames Feist                 }
119783ff9ab6SJames Feist             }
119883ff9ab6SJames Feist         }
11999c310685SBorawski.Lukasz     }
12009c310685SBorawski.Lukasz 
12011abe55efSEd Tanous     std::string getDateTime() const
12021abe55efSEd Tanous     {
12039c310685SBorawski.Lukasz         std::array<char, 128> dateTime;
12049c310685SBorawski.Lukasz         std::string redfishDateTime("0000-00-00T00:00:00Z00:00");
12059c310685SBorawski.Lukasz         std::time_t time = std::time(nullptr);
12069c310685SBorawski.Lukasz 
12079c310685SBorawski.Lukasz         if (std::strftime(dateTime.begin(), dateTime.size(), "%FT%T%z",
12081abe55efSEd Tanous                           std::localtime(&time)))
12091abe55efSEd Tanous         {
12109c310685SBorawski.Lukasz             // insert the colon required by the ISO 8601 standard
12119c310685SBorawski.Lukasz             redfishDateTime = std::string(dateTime.data());
12129c310685SBorawski.Lukasz             redfishDateTime.insert(redfishDateTime.end() - 2, ':');
12139c310685SBorawski.Lukasz         }
12149c310685SBorawski.Lukasz 
12159c310685SBorawski.Lukasz         return redfishDateTime;
12169c310685SBorawski.Lukasz     }
12170f74e643SEd Tanous 
12180f74e643SEd Tanous     std::string uuid;
12199c310685SBorawski.Lukasz };
12209c310685SBorawski.Lukasz 
12211abe55efSEd Tanous class ManagerCollection : public Node
12221abe55efSEd Tanous {
12239c310685SBorawski.Lukasz   public:
12241abe55efSEd Tanous     ManagerCollection(CrowApp& app) : Node(app, "/redfish/v1/Managers/")
12251abe55efSEd Tanous     {
1226a434f2bdSEd Tanous         entityPrivileges = {
1227a434f2bdSEd Tanous             {boost::beast::http::verb::get, {{"Login"}}},
1228e0d918bcSEd Tanous             {boost::beast::http::verb::head, {{"Login"}}},
1229e0d918bcSEd Tanous             {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
1230e0d918bcSEd Tanous             {boost::beast::http::verb::put, {{"ConfigureManager"}}},
1231e0d918bcSEd Tanous             {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
1232e0d918bcSEd Tanous             {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
12339c310685SBorawski.Lukasz     }
12349c310685SBorawski.Lukasz 
12359c310685SBorawski.Lukasz   private:
123655c7b7a2SEd Tanous     void doGet(crow::Response& res, const crow::Request& req,
12371abe55efSEd Tanous                const std::vector<std::string>& params) override
12381abe55efSEd Tanous     {
123983ff9ab6SJames Feist         // Collections don't include the static data added by SubRoute
124083ff9ab6SJames Feist         // because it has a duplicate entry for members
124155c7b7a2SEd Tanous         res.jsonValue["@odata.id"] = "/redfish/v1/Managers";
124255c7b7a2SEd Tanous         res.jsonValue["@odata.type"] = "#ManagerCollection.ManagerCollection";
124355c7b7a2SEd Tanous         res.jsonValue["@odata.context"] =
124455c7b7a2SEd Tanous             "/redfish/v1/$metadata#ManagerCollection.ManagerCollection";
124555c7b7a2SEd Tanous         res.jsonValue["Name"] = "Manager Collection";
124655c7b7a2SEd Tanous         res.jsonValue["Members@odata.count"] = 1;
124755c7b7a2SEd Tanous         res.jsonValue["Members"] = {
12485b4aa86bSJames Feist             {{"@odata.id", "/redfish/v1/Managers/bmc"}}};
12499c310685SBorawski.Lukasz         res.end();
12509c310685SBorawski.Lukasz     }
12519c310685SBorawski.Lukasz };
12529c310685SBorawski.Lukasz } // namespace redfish
1253