xref: /openbmc/bmcweb/features/redfish/lib/managers.hpp (revision 73df0db0daca78a62ef262ec3f933c887e7ceaa1)
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>
21af5d6058SSantosh Puranik #include <boost/date_time.hpp>
225b4aa86bSJames Feist #include <dbus_utility.hpp>
23*73df0db0SJames Feist #include <memory>
24af5d6058SSantosh Puranik #include <sstream>
257bffdb7eSBernard Wong #include <utils/systemd_utils.hpp>
26abf2add6SEd Tanous #include <variant>
275b4aa86bSJames Feist 
281abe55efSEd Tanous namespace redfish
291abe55efSEd Tanous {
30ed5befbdSJennifer Lee 
31ed5befbdSJennifer Lee /**
32ed5befbdSJennifer Lee  * ManagerActionsReset class supports handle POST method for Reset action.
33ed5befbdSJennifer Lee  * The class retrieves and sends data directly to dbus.
34ed5befbdSJennifer Lee  */
35ed5befbdSJennifer Lee class ManagerActionsReset : public Node
36ed5befbdSJennifer Lee {
37ed5befbdSJennifer Lee   public:
38ed5befbdSJennifer Lee     ManagerActionsReset(CrowApp& app) :
39ed5befbdSJennifer Lee         Node(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/")
40ed5befbdSJennifer Lee     {
41ed5befbdSJennifer Lee         entityPrivileges = {
42ed5befbdSJennifer Lee             {boost::beast::http::verb::get, {{"Login"}}},
43ed5befbdSJennifer Lee             {boost::beast::http::verb::head, {{"Login"}}},
44ed5befbdSJennifer Lee             {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
45ed5befbdSJennifer Lee             {boost::beast::http::verb::put, {{"ConfigureManager"}}},
46ed5befbdSJennifer Lee             {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
47ed5befbdSJennifer Lee             {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
48ed5befbdSJennifer Lee     }
49ed5befbdSJennifer Lee 
50ed5befbdSJennifer Lee   private:
51ed5befbdSJennifer Lee     /**
52ed5befbdSJennifer Lee      * Function handles POST method request.
53ed5befbdSJennifer Lee      * Analyzes POST body message before sends Reset request data to dbus.
54ed5befbdSJennifer Lee      * OpenBMC allows for ResetType is GracefulRestart only.
55ed5befbdSJennifer Lee      */
56ed5befbdSJennifer Lee     void doPost(crow::Response& res, const crow::Request& req,
57ed5befbdSJennifer Lee                 const std::vector<std::string>& params) override
58ed5befbdSJennifer Lee     {
59ed5befbdSJennifer Lee         std::string resetType;
60ed5befbdSJennifer Lee 
61ed5befbdSJennifer Lee         if (!json_util::readJson(req, res, "ResetType", resetType))
62ed5befbdSJennifer Lee         {
63ed5befbdSJennifer Lee             return;
64ed5befbdSJennifer Lee         }
65ed5befbdSJennifer Lee 
66ed5befbdSJennifer Lee         if (resetType != "GracefulRestart")
67ed5befbdSJennifer Lee         {
68ed5befbdSJennifer Lee             res.result(boost::beast::http::status::bad_request);
69ed5befbdSJennifer Lee             messages::actionParameterNotSupported(res, resetType, "ResetType");
70ed5befbdSJennifer Lee             BMCWEB_LOG_ERROR << "Request incorrect action parameter: "
71ed5befbdSJennifer Lee                              << resetType;
72ed5befbdSJennifer Lee             res.end();
73ed5befbdSJennifer Lee             return;
74ed5befbdSJennifer Lee         }
75ed5befbdSJennifer Lee         doBMCGracefulRestart(res, req, params);
76ed5befbdSJennifer Lee     }
77ed5befbdSJennifer Lee 
78ed5befbdSJennifer Lee     /**
79ed5befbdSJennifer Lee      * Function transceives data with dbus directly.
80ed5befbdSJennifer Lee      * All BMC state properties will be retrieved before sending reset request.
81ed5befbdSJennifer Lee      */
82ed5befbdSJennifer Lee     void doBMCGracefulRestart(crow::Response& res, const crow::Request& req,
83ed5befbdSJennifer Lee                               const std::vector<std::string>& params)
84ed5befbdSJennifer Lee     {
85ed5befbdSJennifer Lee         const char* processName = "xyz.openbmc_project.State.BMC";
86ed5befbdSJennifer Lee         const char* objectPath = "/xyz/openbmc_project/state/bmc0";
87ed5befbdSJennifer Lee         const char* interfaceName = "xyz.openbmc_project.State.BMC";
88ed5befbdSJennifer Lee         const std::string& propertyValue =
89ed5befbdSJennifer Lee             "xyz.openbmc_project.State.BMC.Transition.Reboot";
90ed5befbdSJennifer Lee         const char* destProperty = "RequestedBMCTransition";
91ed5befbdSJennifer Lee 
92ed5befbdSJennifer Lee         // Create the D-Bus variant for D-Bus call.
93ed5befbdSJennifer Lee         VariantType dbusPropertyValue(propertyValue);
94ed5befbdSJennifer Lee 
95ed5befbdSJennifer Lee         std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res);
96ed5befbdSJennifer Lee 
97ed5befbdSJennifer Lee         crow::connections::systemBus->async_method_call(
98ed5befbdSJennifer Lee             [asyncResp](const boost::system::error_code ec) {
99ed5befbdSJennifer Lee                 // Use "Set" method to set the property value.
100ed5befbdSJennifer Lee                 if (ec)
101ed5befbdSJennifer Lee                 {
102ed5befbdSJennifer Lee                     BMCWEB_LOG_ERROR << "[Set] Bad D-Bus request error: " << ec;
103ed5befbdSJennifer Lee                     messages::internalError(asyncResp->res);
104ed5befbdSJennifer Lee                     return;
105ed5befbdSJennifer Lee                 }
106ed5befbdSJennifer Lee 
107ed5befbdSJennifer Lee                 messages::success(asyncResp->res);
108ed5befbdSJennifer Lee             },
109ed5befbdSJennifer Lee             processName, objectPath, "org.freedesktop.DBus.Properties", "Set",
110ed5befbdSJennifer Lee             interfaceName, destProperty, dbusPropertyValue);
111ed5befbdSJennifer Lee     }
112ed5befbdSJennifer Lee };
113ed5befbdSJennifer Lee 
1145b4aa86bSJames Feist static constexpr const char* objectManagerIface =
1155b4aa86bSJames Feist     "org.freedesktop.DBus.ObjectManager";
1165b4aa86bSJames Feist static constexpr const char* pidConfigurationIface =
1175b4aa86bSJames Feist     "xyz.openbmc_project.Configuration.Pid";
1185b4aa86bSJames Feist static constexpr const char* pidZoneConfigurationIface =
1195b4aa86bSJames Feist     "xyz.openbmc_project.Configuration.Pid.Zone";
120b7a08d04SJames Feist static constexpr const char* stepwiseConfigurationIface =
121b7a08d04SJames Feist     "xyz.openbmc_project.Configuration.Stepwise";
122*73df0db0SJames Feist static constexpr const char* thermalModeIface =
123*73df0db0SJames Feist     "xyz.openbmc_project.Control.ThermalMode";
1249c310685SBorawski.Lukasz 
1255b4aa86bSJames Feist static void asyncPopulatePid(const std::string& connection,
1265b4aa86bSJames Feist                              const std::string& path,
127*73df0db0SJames Feist                              const std::string& currentProfile,
128*73df0db0SJames Feist                              const std::vector<std::string>& supportedProfiles,
1295b4aa86bSJames Feist                              std::shared_ptr<AsyncResp> asyncResp)
1305b4aa86bSJames Feist {
1315b4aa86bSJames Feist 
1325b4aa86bSJames Feist     crow::connections::systemBus->async_method_call(
133*73df0db0SJames Feist         [asyncResp, currentProfile, supportedProfiles](
134*73df0db0SJames Feist             const boost::system::error_code ec,
1355b4aa86bSJames Feist             const dbus::utility::ManagedObjectType& managedObj) {
1365b4aa86bSJames Feist             if (ec)
1375b4aa86bSJames Feist             {
1385b4aa86bSJames Feist                 BMCWEB_LOG_ERROR << ec;
1395b4aa86bSJames Feist                 asyncResp->res.jsonValue.clear();
140f12894f8SJason M. Bills                 messages::internalError(asyncResp->res);
1415b4aa86bSJames Feist                 return;
1425b4aa86bSJames Feist             }
1435b4aa86bSJames Feist             nlohmann::json& configRoot =
1445b4aa86bSJames Feist                 asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"];
1455b4aa86bSJames Feist             nlohmann::json& fans = configRoot["FanControllers"];
1465b4aa86bSJames Feist             fans["@odata.type"] = "#OemManager.FanControllers";
1475b4aa86bSJames Feist             fans["@odata.context"] =
1485b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.FanControllers";
1495b4aa86bSJames Feist             fans["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/"
1505b4aa86bSJames Feist                                 "Fan/FanControllers";
1515b4aa86bSJames Feist 
1525b4aa86bSJames Feist             nlohmann::json& pids = configRoot["PidControllers"];
1535b4aa86bSJames Feist             pids["@odata.type"] = "#OemManager.PidControllers";
1545b4aa86bSJames Feist             pids["@odata.context"] =
1555b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.PidControllers";
1565b4aa86bSJames Feist             pids["@odata.id"] =
1575b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers";
1585b4aa86bSJames Feist 
159b7a08d04SJames Feist             nlohmann::json& stepwise = configRoot["StepwiseControllers"];
160b7a08d04SJames Feist             stepwise["@odata.type"] = "#OemManager.StepwiseControllers";
161b7a08d04SJames Feist             stepwise["@odata.context"] =
162b7a08d04SJames Feist                 "/redfish/v1/$metadata#OemManager.StepwiseControllers";
163b7a08d04SJames Feist             stepwise["@odata.id"] =
164b7a08d04SJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers";
165b7a08d04SJames Feist 
1665b4aa86bSJames Feist             nlohmann::json& zones = configRoot["FanZones"];
1675b4aa86bSJames Feist             zones["@odata.id"] =
1685b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones";
1695b4aa86bSJames Feist             zones["@odata.type"] = "#OemManager.FanZones";
1705b4aa86bSJames Feist             zones["@odata.context"] =
1715b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.FanZones";
1725b4aa86bSJames Feist             configRoot["@odata.id"] =
1735b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan";
1745b4aa86bSJames Feist             configRoot["@odata.type"] = "#OemManager.Fan";
1755b4aa86bSJames Feist             configRoot["@odata.context"] =
1765b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.Fan";
177*73df0db0SJames Feist             configRoot["Profile@Redfish.AllowableValues"] = supportedProfiles;
178*73df0db0SJames Feist 
179*73df0db0SJames Feist             if (!currentProfile.empty())
180*73df0db0SJames Feist             {
181*73df0db0SJames Feist                 configRoot["Profile"] = currentProfile;
182*73df0db0SJames Feist             }
183*73df0db0SJames Feist             BMCWEB_LOG_ERROR << "profile = " << currentProfile << " !";
1845b4aa86bSJames Feist 
1855b4aa86bSJames Feist             for (const auto& pathPair : managedObj)
1865b4aa86bSJames Feist             {
1875b4aa86bSJames Feist                 for (const auto& intfPair : pathPair.second)
1885b4aa86bSJames Feist                 {
1895b4aa86bSJames Feist                     if (intfPair.first != pidConfigurationIface &&
190b7a08d04SJames Feist                         intfPair.first != pidZoneConfigurationIface &&
191b7a08d04SJames Feist                         intfPair.first != stepwiseConfigurationIface)
1925b4aa86bSJames Feist                     {
1935b4aa86bSJames Feist                         continue;
1945b4aa86bSJames Feist                     }
1955b4aa86bSJames Feist                     auto findName = intfPair.second.find("Name");
1965b4aa86bSJames Feist                     if (findName == intfPair.second.end())
1975b4aa86bSJames Feist                     {
1985b4aa86bSJames Feist                         BMCWEB_LOG_ERROR << "Pid Field missing Name";
199a08b46ccSJason M. Bills                         messages::internalError(asyncResp->res);
2005b4aa86bSJames Feist                         return;
2015b4aa86bSJames Feist                     }
202*73df0db0SJames Feist 
2035b4aa86bSJames Feist                     const std::string* namePtr =
204abf2add6SEd Tanous                         std::get_if<std::string>(&findName->second);
2055b4aa86bSJames Feist                     if (namePtr == nullptr)
2065b4aa86bSJames Feist                     {
2075b4aa86bSJames Feist                         BMCWEB_LOG_ERROR << "Pid Name Field illegal";
208b7a08d04SJames Feist                         messages::internalError(asyncResp->res);
2095b4aa86bSJames Feist                         return;
2105b4aa86bSJames Feist                     }
2115b4aa86bSJames Feist                     std::string name = *namePtr;
2125b4aa86bSJames Feist                     dbus::utility::escapePathForDbus(name);
213*73df0db0SJames Feist 
214*73df0db0SJames Feist                     auto findProfiles = intfPair.second.find("Profiles");
215*73df0db0SJames Feist                     if (findProfiles != intfPair.second.end())
216*73df0db0SJames Feist                     {
217*73df0db0SJames Feist                         const std::vector<std::string>* profiles =
218*73df0db0SJames Feist                             std::get_if<std::vector<std::string>>(
219*73df0db0SJames Feist                                 &findProfiles->second);
220*73df0db0SJames Feist                         if (profiles == nullptr)
221*73df0db0SJames Feist                         {
222*73df0db0SJames Feist                             BMCWEB_LOG_ERROR << "Pid Profiles Field illegal";
223*73df0db0SJames Feist                             messages::internalError(asyncResp->res);
224*73df0db0SJames Feist                             return;
225*73df0db0SJames Feist                         }
226*73df0db0SJames Feist                         if (std::find(profiles->begin(), profiles->end(),
227*73df0db0SJames Feist                                       currentProfile) == profiles->end())
228*73df0db0SJames Feist                         {
229*73df0db0SJames Feist                             BMCWEB_LOG_INFO
230*73df0db0SJames Feist                                 << name << " not supported in current profile";
231*73df0db0SJames Feist                             continue;
232*73df0db0SJames Feist                         }
233*73df0db0SJames Feist                     }
234b7a08d04SJames Feist                     nlohmann::json* config = nullptr;
235c33a90ecSJames Feist 
236c33a90ecSJames Feist                     const std::string* classPtr = nullptr;
237c33a90ecSJames Feist                     auto findClass = intfPair.second.find("Class");
238c33a90ecSJames Feist                     if (findClass != intfPair.second.end())
239c33a90ecSJames Feist                     {
240c33a90ecSJames Feist                         classPtr = std::get_if<std::string>(&findClass->second);
241c33a90ecSJames Feist                     }
242c33a90ecSJames Feist 
2435b4aa86bSJames Feist                     if (intfPair.first == pidZoneConfigurationIface)
2445b4aa86bSJames Feist                     {
2455b4aa86bSJames Feist                         std::string chassis;
2465b4aa86bSJames Feist                         if (!dbus::utility::getNthStringFromPath(
2475b4aa86bSJames Feist                                 pathPair.first.str, 5, chassis))
2485b4aa86bSJames Feist                         {
2495b4aa86bSJames Feist                             chassis = "#IllegalValue";
2505b4aa86bSJames Feist                         }
2515b4aa86bSJames Feist                         nlohmann::json& zone = zones[name];
2525b4aa86bSJames Feist                         zone["Chassis"] = {
2535b4aa86bSJames Feist                             {"@odata.id", "/redfish/v1/Chassis/" + chassis}};
2545b4aa86bSJames Feist                         zone["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/"
2555b4aa86bSJames Feist                                             "OpenBmc/Fan/FanZones/" +
2565b4aa86bSJames Feist                                             name;
2575b4aa86bSJames Feist                         zone["@odata.type"] = "#OemManager.FanZone";
2585b4aa86bSJames Feist                         zone["@odata.context"] =
2595b4aa86bSJames Feist                             "/redfish/v1/$metadata#OemManager.FanZone";
260b7a08d04SJames Feist                         config = &zone;
2615b4aa86bSJames Feist                     }
2625b4aa86bSJames Feist 
263b7a08d04SJames Feist                     else if (intfPair.first == stepwiseConfigurationIface)
2645b4aa86bSJames Feist                     {
265c33a90ecSJames Feist                         if (classPtr == nullptr)
266c33a90ecSJames Feist                         {
267c33a90ecSJames Feist                             BMCWEB_LOG_ERROR << "Pid Class Field illegal";
268c33a90ecSJames Feist                             messages::internalError(asyncResp->res);
269c33a90ecSJames Feist                             return;
270c33a90ecSJames Feist                         }
271c33a90ecSJames Feist 
272b7a08d04SJames Feist                         nlohmann::json& controller = stepwise[name];
273b7a08d04SJames Feist                         config = &controller;
2745b4aa86bSJames Feist 
275b7a08d04SJames Feist                         controller["@odata.id"] =
276b7a08d04SJames Feist                             "/redfish/v1/Managers/bmc#/Oem/"
277b7a08d04SJames Feist                             "OpenBmc/Fan/StepwiseControllers/" +
278b7a08d04SJames Feist                             std::string(name);
279b7a08d04SJames Feist                         controller["@odata.type"] =
280b7a08d04SJames Feist                             "#OemManager.StepwiseController";
281b7a08d04SJames Feist 
282b7a08d04SJames Feist                         controller["@odata.context"] =
283b7a08d04SJames Feist                             "/redfish/v1/"
284b7a08d04SJames Feist                             "$metadata#OemManager.StepwiseController";
285c33a90ecSJames Feist                         controller["Direction"] = *classPtr;
2865b4aa86bSJames Feist                     }
2875b4aa86bSJames Feist 
2885b4aa86bSJames Feist                     // pid and fans are off the same configuration
289b7a08d04SJames Feist                     else if (intfPair.first == pidConfigurationIface)
2905b4aa86bSJames Feist                     {
291c33a90ecSJames Feist 
2925b4aa86bSJames Feist                         if (classPtr == nullptr)
2935b4aa86bSJames Feist                         {
2945b4aa86bSJames Feist                             BMCWEB_LOG_ERROR << "Pid Class Field illegal";
295a08b46ccSJason M. Bills                             messages::internalError(asyncResp->res);
2965b4aa86bSJames Feist                             return;
2975b4aa86bSJames Feist                         }
2985b4aa86bSJames Feist                         bool isFan = *classPtr == "fan";
2995b4aa86bSJames Feist                         nlohmann::json& element =
3005b4aa86bSJames Feist                             isFan ? fans[name] : pids[name];
301b7a08d04SJames Feist                         config = &element;
3025b4aa86bSJames Feist                         if (isFan)
3035b4aa86bSJames Feist                         {
3045b4aa86bSJames Feist                             element["@odata.id"] =
3055b4aa86bSJames Feist                                 "/redfish/v1/Managers/bmc#/Oem/"
3065b4aa86bSJames Feist                                 "OpenBmc/Fan/FanControllers/" +
3075b4aa86bSJames Feist                                 std::string(name);
3085b4aa86bSJames Feist                             element["@odata.type"] =
3095b4aa86bSJames Feist                                 "#OemManager.FanController";
3105b4aa86bSJames Feist 
3115b4aa86bSJames Feist                             element["@odata.context"] =
3125b4aa86bSJames Feist                                 "/redfish/v1/"
3135b4aa86bSJames Feist                                 "$metadata#OemManager.FanController";
3145b4aa86bSJames Feist                         }
3155b4aa86bSJames Feist                         else
3165b4aa86bSJames Feist                         {
3175b4aa86bSJames Feist                             element["@odata.id"] =
3185b4aa86bSJames Feist                                 "/redfish/v1/Managers/bmc#/Oem/"
3195b4aa86bSJames Feist                                 "OpenBmc/Fan/PidControllers/" +
3205b4aa86bSJames Feist                                 std::string(name);
3215b4aa86bSJames Feist                             element["@odata.type"] =
3225b4aa86bSJames Feist                                 "#OemManager.PidController";
3235b4aa86bSJames Feist                             element["@odata.context"] =
3245b4aa86bSJames Feist                                 "/redfish/v1/$metadata"
3255b4aa86bSJames Feist                                 "#OemManager.PidController";
3265b4aa86bSJames Feist                         }
327b7a08d04SJames Feist                     }
328b7a08d04SJames Feist                     else
329b7a08d04SJames Feist                     {
330b7a08d04SJames Feist                         BMCWEB_LOG_ERROR << "Unexpected configuration";
331b7a08d04SJames Feist                         messages::internalError(asyncResp->res);
332b7a08d04SJames Feist                         return;
333b7a08d04SJames Feist                     }
334b7a08d04SJames Feist 
335b7a08d04SJames Feist                     // used for making maps out of 2 vectors
336b7a08d04SJames Feist                     const std::vector<double>* keys = nullptr;
337b7a08d04SJames Feist                     const std::vector<double>* values = nullptr;
338b7a08d04SJames Feist 
339b7a08d04SJames Feist                     for (const auto& propertyPair : intfPair.second)
340b7a08d04SJames Feist                     {
341b7a08d04SJames Feist                         if (propertyPair.first == "Type" ||
342b7a08d04SJames Feist                             propertyPair.first == "Class" ||
343b7a08d04SJames Feist                             propertyPair.first == "Name")
344b7a08d04SJames Feist                         {
345b7a08d04SJames Feist                             continue;
346b7a08d04SJames Feist                         }
347b7a08d04SJames Feist 
348b7a08d04SJames Feist                         // zones
349b7a08d04SJames Feist                         if (intfPair.first == pidZoneConfigurationIface)
350b7a08d04SJames Feist                         {
351b7a08d04SJames Feist                             const double* ptr =
352abf2add6SEd Tanous                                 std::get_if<double>(&propertyPair.second);
353b7a08d04SJames Feist                             if (ptr == nullptr)
354b7a08d04SJames Feist                             {
355b7a08d04SJames Feist                                 BMCWEB_LOG_ERROR << "Field Illegal "
356b7a08d04SJames Feist                                                  << propertyPair.first;
357b7a08d04SJames Feist                                 messages::internalError(asyncResp->res);
358b7a08d04SJames Feist                                 return;
359b7a08d04SJames Feist                             }
360b7a08d04SJames Feist                             (*config)[propertyPair.first] = *ptr;
361b7a08d04SJames Feist                         }
362b7a08d04SJames Feist 
363b7a08d04SJames Feist                         if (intfPair.first == stepwiseConfigurationIface)
364b7a08d04SJames Feist                         {
365b7a08d04SJames Feist                             if (propertyPair.first == "Reading" ||
366b7a08d04SJames Feist                                 propertyPair.first == "Output")
367b7a08d04SJames Feist                             {
368b7a08d04SJames Feist                                 const std::vector<double>* ptr =
369abf2add6SEd Tanous                                     std::get_if<std::vector<double>>(
370b7a08d04SJames Feist                                         &propertyPair.second);
371b7a08d04SJames Feist 
372b7a08d04SJames Feist                                 if (ptr == nullptr)
373b7a08d04SJames Feist                                 {
374b7a08d04SJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
375b7a08d04SJames Feist                                                      << propertyPair.first;
376b7a08d04SJames Feist                                     messages::internalError(asyncResp->res);
377b7a08d04SJames Feist                                     return;
378b7a08d04SJames Feist                                 }
379b7a08d04SJames Feist 
380b7a08d04SJames Feist                                 if (propertyPair.first == "Reading")
381b7a08d04SJames Feist                                 {
382b7a08d04SJames Feist                                     keys = ptr;
383b7a08d04SJames Feist                                 }
384b7a08d04SJames Feist                                 else
385b7a08d04SJames Feist                                 {
386b7a08d04SJames Feist                                     values = ptr;
387b7a08d04SJames Feist                                 }
388b7a08d04SJames Feist                                 if (keys && values)
389b7a08d04SJames Feist                                 {
390b7a08d04SJames Feist                                     if (keys->size() != values->size())
391b7a08d04SJames Feist                                     {
392b7a08d04SJames Feist                                         BMCWEB_LOG_ERROR
393b7a08d04SJames Feist                                             << "Reading and Output size don't "
394b7a08d04SJames Feist                                                "match ";
395b7a08d04SJames Feist                                         messages::internalError(asyncResp->res);
396b7a08d04SJames Feist                                         return;
397b7a08d04SJames Feist                                     }
398b7a08d04SJames Feist                                     nlohmann::json& steps = (*config)["Steps"];
399b7a08d04SJames Feist                                     steps = nlohmann::json::array();
400b7a08d04SJames Feist                                     for (size_t ii = 0; ii < keys->size(); ii++)
401b7a08d04SJames Feist                                     {
402b7a08d04SJames Feist                                         steps.push_back(
403b7a08d04SJames Feist                                             {{"Target", (*keys)[ii]},
404b7a08d04SJames Feist                                              {"Output", (*values)[ii]}});
405b7a08d04SJames Feist                                     }
406b7a08d04SJames Feist                                 }
407b7a08d04SJames Feist                             }
408b7a08d04SJames Feist                             if (propertyPair.first == "NegativeHysteresis" ||
409b7a08d04SJames Feist                                 propertyPair.first == "PositiveHysteresis")
410b7a08d04SJames Feist                             {
411b7a08d04SJames Feist                                 const double* ptr =
412abf2add6SEd Tanous                                     std::get_if<double>(&propertyPair.second);
413b7a08d04SJames Feist                                 if (ptr == nullptr)
414b7a08d04SJames Feist                                 {
415b7a08d04SJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
416b7a08d04SJames Feist                                                      << propertyPair.first;
417b7a08d04SJames Feist                                     messages::internalError(asyncResp->res);
418b7a08d04SJames Feist                                     return;
419b7a08d04SJames Feist                                 }
420b7a08d04SJames Feist                                 (*config)[propertyPair.first] = *ptr;
421b7a08d04SJames Feist                             }
422b7a08d04SJames Feist                         }
423b7a08d04SJames Feist 
424b7a08d04SJames Feist                         // pid and fans are off the same configuration
425b7a08d04SJames Feist                         if (intfPair.first == pidConfigurationIface ||
426b7a08d04SJames Feist                             intfPair.first == stepwiseConfigurationIface)
427b7a08d04SJames Feist                         {
4285b4aa86bSJames Feist 
4295b4aa86bSJames Feist                             if (propertyPair.first == "Zones")
4305b4aa86bSJames Feist                             {
4315b4aa86bSJames Feist                                 const std::vector<std::string>* inputs =
432abf2add6SEd Tanous                                     std::get_if<std::vector<std::string>>(
4331b6b96c5SEd Tanous                                         &propertyPair.second);
4345b4aa86bSJames Feist 
4355b4aa86bSJames Feist                                 if (inputs == nullptr)
4365b4aa86bSJames Feist                                 {
4375b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR
4385b4aa86bSJames Feist                                         << "Zones Pid Field Illegal";
439a08b46ccSJason M. Bills                                     messages::internalError(asyncResp->res);
4405b4aa86bSJames Feist                                     return;
4415b4aa86bSJames Feist                                 }
442b7a08d04SJames Feist                                 auto& data = (*config)[propertyPair.first];
4435b4aa86bSJames Feist                                 data = nlohmann::json::array();
4445b4aa86bSJames Feist                                 for (std::string itemCopy : *inputs)
4455b4aa86bSJames Feist                                 {
4465b4aa86bSJames Feist                                     dbus::utility::escapePathForDbus(itemCopy);
4475b4aa86bSJames Feist                                     data.push_back(
4485b4aa86bSJames Feist                                         {{"@odata.id",
4495b4aa86bSJames Feist                                           "/redfish/v1/Managers/bmc#/Oem/"
4505b4aa86bSJames Feist                                           "OpenBmc/Fan/FanZones/" +
4515b4aa86bSJames Feist                                               itemCopy}});
4525b4aa86bSJames Feist                                 }
4535b4aa86bSJames Feist                             }
4545b4aa86bSJames Feist                             // todo(james): may never happen, but this
4555b4aa86bSJames Feist                             // assumes configuration data referenced in the
4565b4aa86bSJames Feist                             // PID config is provided by the same daemon, we
4575b4aa86bSJames Feist                             // could add another loop to cover all cases,
4585b4aa86bSJames Feist                             // but I'm okay kicking this can down the road a
4595b4aa86bSJames Feist                             // bit
4605b4aa86bSJames Feist 
4615b4aa86bSJames Feist                             else if (propertyPair.first == "Inputs" ||
4625b4aa86bSJames Feist                                      propertyPair.first == "Outputs")
4635b4aa86bSJames Feist                             {
464b7a08d04SJames Feist                                 auto& data = (*config)[propertyPair.first];
4655b4aa86bSJames Feist                                 const std::vector<std::string>* inputs =
466abf2add6SEd Tanous                                     std::get_if<std::vector<std::string>>(
4671b6b96c5SEd Tanous                                         &propertyPair.second);
4685b4aa86bSJames Feist 
4695b4aa86bSJames Feist                                 if (inputs == nullptr)
4705b4aa86bSJames Feist                                 {
4715b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
4725b4aa86bSJames Feist                                                      << propertyPair.first;
473f12894f8SJason M. Bills                                     messages::internalError(asyncResp->res);
4745b4aa86bSJames Feist                                     return;
4755b4aa86bSJames Feist                                 }
4765b4aa86bSJames Feist                                 data = *inputs;
4775b4aa86bSJames Feist                             } // doubles
4785b4aa86bSJames Feist                             else if (propertyPair.first ==
4795b4aa86bSJames Feist                                          "FFGainCoefficient" ||
4805b4aa86bSJames Feist                                      propertyPair.first == "FFOffCoefficient" ||
4815b4aa86bSJames Feist                                      propertyPair.first == "ICoefficient" ||
4825b4aa86bSJames Feist                                      propertyPair.first == "ILimitMax" ||
4835b4aa86bSJames Feist                                      propertyPair.first == "ILimitMin" ||
484aad1a257SJames Feist                                      propertyPair.first ==
485aad1a257SJames Feist                                          "PositiveHysteresis" ||
486aad1a257SJames Feist                                      propertyPair.first ==
487aad1a257SJames Feist                                          "NegativeHysteresis" ||
4885b4aa86bSJames Feist                                      propertyPair.first == "OutLimitMax" ||
4895b4aa86bSJames Feist                                      propertyPair.first == "OutLimitMin" ||
4905b4aa86bSJames Feist                                      propertyPair.first == "PCoefficient" ||
4917625cb81SJames Feist                                      propertyPair.first == "SetPoint" ||
4925b4aa86bSJames Feist                                      propertyPair.first == "SlewNeg" ||
4935b4aa86bSJames Feist                                      propertyPair.first == "SlewPos")
4945b4aa86bSJames Feist                             {
4955b4aa86bSJames Feist                                 const double* ptr =
496abf2add6SEd Tanous                                     std::get_if<double>(&propertyPair.second);
4975b4aa86bSJames Feist                                 if (ptr == nullptr)
4985b4aa86bSJames Feist                                 {
4995b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
5005b4aa86bSJames Feist                                                      << propertyPair.first;
501f12894f8SJason M. Bills                                     messages::internalError(asyncResp->res);
5025b4aa86bSJames Feist                                     return;
5035b4aa86bSJames Feist                                 }
504b7a08d04SJames Feist                                 (*config)[propertyPair.first] = *ptr;
5055b4aa86bSJames Feist                             }
5065b4aa86bSJames Feist                         }
5075b4aa86bSJames Feist                     }
5085b4aa86bSJames Feist                 }
5095b4aa86bSJames Feist             }
5105b4aa86bSJames Feist         },
5115b4aa86bSJames Feist         connection, path, objectManagerIface, "GetManagedObjects");
5125b4aa86bSJames Feist }
513ca537928SJennifer Lee 
51483ff9ab6SJames Feist enum class CreatePIDRet
51583ff9ab6SJames Feist {
51683ff9ab6SJames Feist     fail,
51783ff9ab6SJames Feist     del,
51883ff9ab6SJames Feist     patch
51983ff9ab6SJames Feist };
52083ff9ab6SJames Feist 
5215f2caaefSJames Feist static bool getZonesFromJsonReq(const std::shared_ptr<AsyncResp>& response,
5225f2caaefSJames Feist                                 std::vector<nlohmann::json>& config,
5235f2caaefSJames Feist                                 std::vector<std::string>& zones)
5245f2caaefSJames Feist {
525b6baeaa4SJames Feist     if (config.empty())
526b6baeaa4SJames Feist     {
527b6baeaa4SJames Feist         BMCWEB_LOG_ERROR << "Empty Zones";
528b6baeaa4SJames Feist         messages::propertyValueFormatError(response->res,
529b6baeaa4SJames Feist                                            nlohmann::json::array(), "Zones");
530b6baeaa4SJames Feist         return false;
531b6baeaa4SJames Feist     }
5325f2caaefSJames Feist     for (auto& odata : config)
5335f2caaefSJames Feist     {
5345f2caaefSJames Feist         std::string path;
5355f2caaefSJames Feist         if (!redfish::json_util::readJson(odata, response->res, "@odata.id",
5365f2caaefSJames Feist                                           path))
5375f2caaefSJames Feist         {
5385f2caaefSJames Feist             return false;
5395f2caaefSJames Feist         }
5405f2caaefSJames Feist         std::string input;
54161adbda3SJames Feist 
54261adbda3SJames Feist         // 8 below comes from
54361adbda3SJames Feist         // /redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/Left
54461adbda3SJames Feist         //     0    1     2      3    4    5      6     7      8
54561adbda3SJames Feist         if (!dbus::utility::getNthStringFromPath(path, 8, input))
5465f2caaefSJames Feist         {
5475f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Got invalid path " << path;
5485f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Illegal Type Zones";
5495f2caaefSJames Feist             messages::propertyValueFormatError(response->res, odata.dump(),
5505f2caaefSJames Feist                                                "Zones");
5515f2caaefSJames Feist             return false;
5525f2caaefSJames Feist         }
5535f2caaefSJames Feist         boost::replace_all(input, "_", " ");
5545f2caaefSJames Feist         zones.emplace_back(std::move(input));
5555f2caaefSJames Feist     }
5565f2caaefSJames Feist     return true;
5575f2caaefSJames Feist }
5585f2caaefSJames Feist 
559*73df0db0SJames Feist static const dbus::utility::ManagedItem*
560*73df0db0SJames Feist     findChassis(const dbus::utility::ManagedObjectType& managedObj,
561b6baeaa4SJames Feist                 const std::string& value, std::string& chassis)
562b6baeaa4SJames Feist {
563b6baeaa4SJames Feist     BMCWEB_LOG_DEBUG << "Find Chassis: " << value << "\n";
564b6baeaa4SJames Feist 
565b6baeaa4SJames Feist     std::string escaped = boost::replace_all_copy(value, " ", "_");
566b6baeaa4SJames Feist     escaped = "/" + escaped;
567b6baeaa4SJames Feist     auto it = std::find_if(
568b6baeaa4SJames Feist         managedObj.begin(), managedObj.end(), [&escaped](const auto& obj) {
569b6baeaa4SJames Feist             if (boost::algorithm::ends_with(obj.first.str, escaped))
570b6baeaa4SJames Feist             {
571b6baeaa4SJames Feist                 BMCWEB_LOG_DEBUG << "Matched " << obj.first.str << "\n";
572b6baeaa4SJames Feist                 return true;
573b6baeaa4SJames Feist             }
574b6baeaa4SJames Feist             return false;
575b6baeaa4SJames Feist         });
576b6baeaa4SJames Feist 
577b6baeaa4SJames Feist     if (it == managedObj.end())
578b6baeaa4SJames Feist     {
579*73df0db0SJames Feist         return nullptr;
580b6baeaa4SJames Feist     }
581b6baeaa4SJames Feist     // 5 comes from <chassis-name> being the 5th element
582b6baeaa4SJames Feist     // /xyz/openbmc_project/inventory/system/chassis/<chassis-name>
583*73df0db0SJames Feist     if (dbus::utility::getNthStringFromPath(it->first.str, 5, chassis))
584*73df0db0SJames Feist     {
585*73df0db0SJames Feist         return &(*it);
586*73df0db0SJames Feist     }
587*73df0db0SJames Feist 
588*73df0db0SJames Feist     return nullptr;
589b6baeaa4SJames Feist }
590b6baeaa4SJames Feist 
59183ff9ab6SJames Feist static CreatePIDRet createPidInterface(
59283ff9ab6SJames Feist     const std::shared_ptr<AsyncResp>& response, const std::string& type,
593b6baeaa4SJames Feist     nlohmann::json::iterator it, const std::string& path,
59483ff9ab6SJames Feist     const dbus::utility::ManagedObjectType& managedObj, bool createNewObject,
59583ff9ab6SJames Feist     boost::container::flat_map<std::string, dbus::utility::DbusVariantType>&
59683ff9ab6SJames Feist         output,
597*73df0db0SJames Feist     std::string& chassis, const std::string& profile)
59883ff9ab6SJames Feist {
59983ff9ab6SJames Feist 
6005f2caaefSJames Feist     // common deleter
601b6baeaa4SJames Feist     if (it.value() == nullptr)
6025f2caaefSJames Feist     {
6035f2caaefSJames Feist         std::string iface;
6045f2caaefSJames Feist         if (type == "PidControllers" || type == "FanControllers")
6055f2caaefSJames Feist         {
6065f2caaefSJames Feist             iface = pidConfigurationIface;
6075f2caaefSJames Feist         }
6085f2caaefSJames Feist         else if (type == "FanZones")
6095f2caaefSJames Feist         {
6105f2caaefSJames Feist             iface = pidZoneConfigurationIface;
6115f2caaefSJames Feist         }
6125f2caaefSJames Feist         else if (type == "StepwiseControllers")
6135f2caaefSJames Feist         {
6145f2caaefSJames Feist             iface = stepwiseConfigurationIface;
6155f2caaefSJames Feist         }
6165f2caaefSJames Feist         else
6175f2caaefSJames Feist         {
6185f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Type "
6195f2caaefSJames Feist                              << type;
6205f2caaefSJames Feist             messages::propertyUnknown(response->res, type);
6215f2caaefSJames Feist             return CreatePIDRet::fail;
6225f2caaefSJames Feist         }
6235f2caaefSJames Feist         // delete interface
6245f2caaefSJames Feist         crow::connections::systemBus->async_method_call(
6255f2caaefSJames Feist             [response, path](const boost::system::error_code ec) {
6265f2caaefSJames Feist                 if (ec)
6275f2caaefSJames Feist                 {
6285f2caaefSJames Feist                     BMCWEB_LOG_ERROR << "Error patching " << path << ": " << ec;
6295f2caaefSJames Feist                     messages::internalError(response->res);
630b6baeaa4SJames Feist                     return;
6315f2caaefSJames Feist                 }
632b6baeaa4SJames Feist                 messages::success(response->res);
6335f2caaefSJames Feist             },
6345f2caaefSJames Feist             "xyz.openbmc_project.EntityManager", path, iface, "Delete");
6355f2caaefSJames Feist         return CreatePIDRet::del;
6365f2caaefSJames Feist     }
6375f2caaefSJames Feist 
638*73df0db0SJames Feist     const dbus::utility::ManagedItem* managedItem = nullptr;
639b6baeaa4SJames Feist     if (!createNewObject)
640b6baeaa4SJames Feist     {
641b6baeaa4SJames Feist         // if we aren't creating a new object, we should be able to find it on
642b6baeaa4SJames Feist         // d-bus
643*73df0db0SJames Feist         managedItem = findChassis(managedObj, it.key(), chassis);
644*73df0db0SJames Feist         if (managedItem == nullptr)
645b6baeaa4SJames Feist         {
646b6baeaa4SJames Feist             BMCWEB_LOG_ERROR << "Failed to get chassis from config patch";
647b6baeaa4SJames Feist             messages::invalidObject(response->res, it.key());
648b6baeaa4SJames Feist             return CreatePIDRet::fail;
649b6baeaa4SJames Feist         }
650b6baeaa4SJames Feist     }
651b6baeaa4SJames Feist 
652*73df0db0SJames Feist     if (profile.size() &&
653*73df0db0SJames Feist         (type == "PidControllers" || type == "FanControllers" ||
654*73df0db0SJames Feist          type == "StepwiseControllers"))
655*73df0db0SJames Feist     {
656*73df0db0SJames Feist         if (managedItem == nullptr)
657*73df0db0SJames Feist         {
658*73df0db0SJames Feist             output["Profiles"] = std::vector<std::string>{profile};
659*73df0db0SJames Feist         }
660*73df0db0SJames Feist         else
661*73df0db0SJames Feist         {
662*73df0db0SJames Feist             std::string interface;
663*73df0db0SJames Feist             if (type == "StepwiseControllers")
664*73df0db0SJames Feist             {
665*73df0db0SJames Feist                 interface = stepwiseConfigurationIface;
666*73df0db0SJames Feist             }
667*73df0db0SJames Feist             else
668*73df0db0SJames Feist             {
669*73df0db0SJames Feist                 interface = pidConfigurationIface;
670*73df0db0SJames Feist             }
671*73df0db0SJames Feist             auto findConfig = managedItem->second.find(interface);
672*73df0db0SJames Feist             if (findConfig == managedItem->second.end())
673*73df0db0SJames Feist             {
674*73df0db0SJames Feist                 BMCWEB_LOG_ERROR
675*73df0db0SJames Feist                     << "Failed to find interface in managed object";
676*73df0db0SJames Feist                 messages::internalError(response->res);
677*73df0db0SJames Feist                 return CreatePIDRet::fail;
678*73df0db0SJames Feist             }
679*73df0db0SJames Feist             auto findProfiles = findConfig->second.find("Profiles");
680*73df0db0SJames Feist             if (findProfiles != findConfig->second.end())
681*73df0db0SJames Feist             {
682*73df0db0SJames Feist                 const std::vector<std::string>* curProfiles =
683*73df0db0SJames Feist                     std::get_if<std::vector<std::string>>(
684*73df0db0SJames Feist                         &(findProfiles->second));
685*73df0db0SJames Feist                 if (curProfiles == nullptr)
686*73df0db0SJames Feist                 {
687*73df0db0SJames Feist                     BMCWEB_LOG_ERROR << "Illegal profiles in managed object";
688*73df0db0SJames Feist                     messages::internalError(response->res);
689*73df0db0SJames Feist                     return CreatePIDRet::fail;
690*73df0db0SJames Feist                 }
691*73df0db0SJames Feist                 if (std::find(curProfiles->begin(), curProfiles->end(),
692*73df0db0SJames Feist                               profile) == curProfiles->end())
693*73df0db0SJames Feist                 {
694*73df0db0SJames Feist                     std::vector<std::string> newProfiles = *curProfiles;
695*73df0db0SJames Feist                     newProfiles.push_back(profile);
696*73df0db0SJames Feist                     output["Profiles"] = newProfiles;
697*73df0db0SJames Feist                 }
698*73df0db0SJames Feist             }
699*73df0db0SJames Feist         }
700*73df0db0SJames Feist     }
701*73df0db0SJames Feist 
70283ff9ab6SJames Feist     if (type == "PidControllers" || type == "FanControllers")
70383ff9ab6SJames Feist     {
70483ff9ab6SJames Feist         if (createNewObject)
70583ff9ab6SJames Feist         {
70683ff9ab6SJames Feist             output["Class"] = type == "PidControllers" ? std::string("temp")
70783ff9ab6SJames Feist                                                        : std::string("fan");
70883ff9ab6SJames Feist             output["Type"] = std::string("Pid");
70983ff9ab6SJames Feist         }
7105f2caaefSJames Feist 
7115f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> zones;
7125f2caaefSJames Feist         std::optional<std::vector<std::string>> inputs;
7135f2caaefSJames Feist         std::optional<std::vector<std::string>> outputs;
7145f2caaefSJames Feist         std::map<std::string, std::optional<double>> doubles;
7155f2caaefSJames Feist         if (!redfish::json_util::readJson(
716b6baeaa4SJames Feist                 it.value(), response->res, "Inputs", inputs, "Outputs", outputs,
7175f2caaefSJames Feist                 "Zones", zones, "FFGainCoefficient",
7185f2caaefSJames Feist                 doubles["FFGainCoefficient"], "FFOffCoefficient",
7195f2caaefSJames Feist                 doubles["FFOffCoefficient"], "ICoefficient",
7205f2caaefSJames Feist                 doubles["ICoefficient"], "ILimitMax", doubles["ILimitMax"],
7215f2caaefSJames Feist                 "ILimitMin", doubles["ILimitMin"], "OutLimitMax",
7225f2caaefSJames Feist                 doubles["OutLimitMax"], "OutLimitMin", doubles["OutLimitMin"],
7235f2caaefSJames Feist                 "PCoefficient", doubles["PCoefficient"], "SetPoint",
7245f2caaefSJames Feist                 doubles["SetPoint"], "SlewNeg", doubles["SlewNeg"], "SlewPos",
725aad1a257SJames Feist                 doubles["SlewPos"], "PositiveHysteresis",
726aad1a257SJames Feist                 doubles["PositiveHysteresis"], "NegativeHysteresis",
727aad1a257SJames Feist                 doubles["NegativeHysteresis"]))
72883ff9ab6SJames Feist         {
7295f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property "
730b6baeaa4SJames Feist                              << it.value().dump();
7315f2caaefSJames Feist             return CreatePIDRet::fail;
73283ff9ab6SJames Feist         }
7335f2caaefSJames Feist         if (zones)
7345f2caaefSJames Feist         {
7355f2caaefSJames Feist             std::vector<std::string> zonesStr;
7365f2caaefSJames Feist             if (!getZonesFromJsonReq(response, *zones, zonesStr))
7375f2caaefSJames Feist             {
7385f2caaefSJames Feist                 BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Zones";
7395f2caaefSJames Feist                 return CreatePIDRet::fail;
7405f2caaefSJames Feist             }
741b6baeaa4SJames Feist             if (chassis.empty() &&
742b6baeaa4SJames Feist                 !findChassis(managedObj, zonesStr[0], chassis))
743b6baeaa4SJames Feist             {
744b6baeaa4SJames Feist                 BMCWEB_LOG_ERROR << "Failed to get chassis from config patch";
745b6baeaa4SJames Feist                 messages::invalidObject(response->res, it.key());
746b6baeaa4SJames Feist                 return CreatePIDRet::fail;
747b6baeaa4SJames Feist             }
748b6baeaa4SJames Feist 
7495f2caaefSJames Feist             output["Zones"] = std::move(zonesStr);
7505f2caaefSJames Feist         }
7515f2caaefSJames Feist         if (inputs || outputs)
7525f2caaefSJames Feist         {
7535f2caaefSJames Feist             std::array<std::optional<std::vector<std::string>>*, 2> containers =
7545f2caaefSJames Feist                 {&inputs, &outputs};
7555f2caaefSJames Feist             size_t index = 0;
7565f2caaefSJames Feist             for (const auto& containerPtr : containers)
7575f2caaefSJames Feist             {
7585f2caaefSJames Feist                 std::optional<std::vector<std::string>>& container =
7595f2caaefSJames Feist                     *containerPtr;
7605f2caaefSJames Feist                 if (!container)
7615f2caaefSJames Feist                 {
7625f2caaefSJames Feist                     index++;
7635f2caaefSJames Feist                     continue;
76483ff9ab6SJames Feist                 }
76583ff9ab6SJames Feist 
7665f2caaefSJames Feist                 for (std::string& value : *container)
76783ff9ab6SJames Feist                 {
7685f2caaefSJames Feist                     boost::replace_all(value, "_", " ");
76983ff9ab6SJames Feist                 }
7705f2caaefSJames Feist                 std::string key;
7715f2caaefSJames Feist                 if (index == 0)
7725f2caaefSJames Feist                 {
7735f2caaefSJames Feist                     key = "Inputs";
7745f2caaefSJames Feist                 }
7755f2caaefSJames Feist                 else
7765f2caaefSJames Feist                 {
7775f2caaefSJames Feist                     key = "Outputs";
7785f2caaefSJames Feist                 }
7795f2caaefSJames Feist                 output[key] = *container;
7805f2caaefSJames Feist                 index++;
7815f2caaefSJames Feist             }
78283ff9ab6SJames Feist         }
78383ff9ab6SJames Feist 
78483ff9ab6SJames Feist         // doubles
7855f2caaefSJames Feist         for (const auto& pairs : doubles)
78683ff9ab6SJames Feist         {
7875f2caaefSJames Feist             if (!pairs.second)
78883ff9ab6SJames Feist             {
7895f2caaefSJames Feist                 continue;
79083ff9ab6SJames Feist             }
7915f2caaefSJames Feist             BMCWEB_LOG_DEBUG << pairs.first << " = " << *pairs.second;
7925f2caaefSJames Feist             output[pairs.first] = *(pairs.second);
7935f2caaefSJames Feist         }
79483ff9ab6SJames Feist     }
79583ff9ab6SJames Feist 
79683ff9ab6SJames Feist     else if (type == "FanZones")
79783ff9ab6SJames Feist     {
79883ff9ab6SJames Feist         output["Type"] = std::string("Pid.Zone");
79983ff9ab6SJames Feist 
8005f2caaefSJames Feist         std::optional<nlohmann::json> chassisContainer;
8015f2caaefSJames Feist         std::optional<double> failSafePercent;
802d3ec07f8SJames Feist         std::optional<double> minThermalOutput;
803b6baeaa4SJames Feist         if (!redfish::json_util::readJson(it.value(), response->res, "Chassis",
8045f2caaefSJames Feist                                           chassisContainer, "FailSafePercent",
805d3ec07f8SJames Feist                                           failSafePercent, "MinThermalOutput",
806d3ec07f8SJames Feist                                           minThermalOutput))
80783ff9ab6SJames Feist         {
8085f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property "
809b6baeaa4SJames Feist                              << it.value().dump();
81083ff9ab6SJames Feist             return CreatePIDRet::fail;
81183ff9ab6SJames Feist         }
8125f2caaefSJames Feist 
8135f2caaefSJames Feist         if (chassisContainer)
81483ff9ab6SJames Feist         {
8155f2caaefSJames Feist 
8165f2caaefSJames Feist             std::string chassisId;
8175f2caaefSJames Feist             if (!redfish::json_util::readJson(*chassisContainer, response->res,
8185f2caaefSJames Feist                                               "@odata.id", chassisId))
8195f2caaefSJames Feist             {
8205f2caaefSJames Feist                 BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property "
8215f2caaefSJames Feist                                  << chassisContainer->dump();
82283ff9ab6SJames Feist                 return CreatePIDRet::fail;
82383ff9ab6SJames Feist             }
82483ff9ab6SJames Feist 
82583ff9ab6SJames Feist             // /refish/v1/chassis/chassis_name/
8265f2caaefSJames Feist             if (!dbus::utility::getNthStringFromPath(chassisId, 3, chassis))
82783ff9ab6SJames Feist             {
8285f2caaefSJames Feist                 BMCWEB_LOG_ERROR << "Got invalid path " << chassisId;
8295f2caaefSJames Feist                 messages::invalidObject(response->res, chassisId);
83083ff9ab6SJames Feist                 return CreatePIDRet::fail;
83183ff9ab6SJames Feist             }
83283ff9ab6SJames Feist         }
833d3ec07f8SJames Feist         if (minThermalOutput)
83483ff9ab6SJames Feist         {
835d3ec07f8SJames Feist             output["MinThermalOutput"] = *minThermalOutput;
8365f2caaefSJames Feist         }
8375f2caaefSJames Feist         if (failSafePercent)
83883ff9ab6SJames Feist         {
8395f2caaefSJames Feist             output["FailSafePercent"] = *failSafePercent;
8405f2caaefSJames Feist         }
8415f2caaefSJames Feist     }
8425f2caaefSJames Feist     else if (type == "StepwiseControllers")
8435f2caaefSJames Feist     {
8445f2caaefSJames Feist         output["Type"] = std::string("Stepwise");
8455f2caaefSJames Feist 
8465f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> zones;
8475f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> steps;
8485f2caaefSJames Feist         std::optional<std::vector<std::string>> inputs;
8495f2caaefSJames Feist         std::optional<double> positiveHysteresis;
8505f2caaefSJames Feist         std::optional<double> negativeHysteresis;
851c33a90ecSJames Feist         std::optional<std::string> direction; // upper clipping curve vs lower
8525f2caaefSJames Feist         if (!redfish::json_util::readJson(
853b6baeaa4SJames Feist                 it.value(), response->res, "Zones", zones, "Steps", steps,
854b6baeaa4SJames Feist                 "Inputs", inputs, "PositiveHysteresis", positiveHysteresis,
855c33a90ecSJames Feist                 "NegativeHysteresis", negativeHysteresis, "Direction",
856c33a90ecSJames Feist                 direction))
8575f2caaefSJames Feist         {
8585f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property "
859b6baeaa4SJames Feist                              << it.value().dump();
86083ff9ab6SJames Feist             return CreatePIDRet::fail;
86183ff9ab6SJames Feist         }
8625f2caaefSJames Feist 
8635f2caaefSJames Feist         if (zones)
86483ff9ab6SJames Feist         {
865b6baeaa4SJames Feist             std::vector<std::string> zonesStrs;
866b6baeaa4SJames Feist             if (!getZonesFromJsonReq(response, *zones, zonesStrs))
8675f2caaefSJames Feist             {
8685f2caaefSJames Feist                 BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Zones";
86983ff9ab6SJames Feist                 return CreatePIDRet::fail;
87083ff9ab6SJames Feist             }
871b6baeaa4SJames Feist             if (chassis.empty() &&
872b6baeaa4SJames Feist                 !findChassis(managedObj, zonesStrs[0], chassis))
873b6baeaa4SJames Feist             {
874b6baeaa4SJames Feist                 BMCWEB_LOG_ERROR << "Failed to get chassis from config patch";
875b6baeaa4SJames Feist                 messages::invalidObject(response->res, it.key());
876b6baeaa4SJames Feist                 return CreatePIDRet::fail;
877b6baeaa4SJames Feist             }
878b6baeaa4SJames Feist             output["Zones"] = std::move(zonesStrs);
8795f2caaefSJames Feist         }
8805f2caaefSJames Feist         if (steps)
8815f2caaefSJames Feist         {
8825f2caaefSJames Feist             std::vector<double> readings;
8835f2caaefSJames Feist             std::vector<double> outputs;
8845f2caaefSJames Feist             for (auto& step : *steps)
8855f2caaefSJames Feist             {
8865f2caaefSJames Feist                 double target;
887b01bf299SEd Tanous                 double output;
8885f2caaefSJames Feist 
8895f2caaefSJames Feist                 if (!redfish::json_util::readJson(step, response->res, "Target",
890b01bf299SEd Tanous                                                   target, "Output", output))
8915f2caaefSJames Feist                 {
8925f2caaefSJames Feist                     BMCWEB_LOG_ERROR << "Line:" << __LINE__
893b6baeaa4SJames Feist                                      << ", Illegal Property "
894b6baeaa4SJames Feist                                      << it.value().dump();
8955f2caaefSJames Feist                     return CreatePIDRet::fail;
8965f2caaefSJames Feist                 }
8975f2caaefSJames Feist                 readings.emplace_back(target);
898b01bf299SEd Tanous                 outputs.emplace_back(output);
8995f2caaefSJames Feist             }
9005f2caaefSJames Feist             output["Reading"] = std::move(readings);
9015f2caaefSJames Feist             output["Output"] = std::move(outputs);
9025f2caaefSJames Feist         }
9035f2caaefSJames Feist         if (inputs)
9045f2caaefSJames Feist         {
9055f2caaefSJames Feist             for (std::string& value : *inputs)
9065f2caaefSJames Feist             {
9075f2caaefSJames Feist                 boost::replace_all(value, "_", " ");
9085f2caaefSJames Feist             }
9095f2caaefSJames Feist             output["Inputs"] = std::move(*inputs);
9105f2caaefSJames Feist         }
9115f2caaefSJames Feist         if (negativeHysteresis)
9125f2caaefSJames Feist         {
9135f2caaefSJames Feist             output["NegativeHysteresis"] = *negativeHysteresis;
9145f2caaefSJames Feist         }
9155f2caaefSJames Feist         if (positiveHysteresis)
9165f2caaefSJames Feist         {
9175f2caaefSJames Feist             output["PositiveHysteresis"] = *positiveHysteresis;
91883ff9ab6SJames Feist         }
919c33a90ecSJames Feist         if (direction)
920c33a90ecSJames Feist         {
921c33a90ecSJames Feist             constexpr const std::array<const char*, 2> allowedDirections = {
922c33a90ecSJames Feist                 "Ceiling", "Floor"};
923c33a90ecSJames Feist             if (std::find(allowedDirections.begin(), allowedDirections.end(),
924c33a90ecSJames Feist                           *direction) == allowedDirections.end())
925c33a90ecSJames Feist             {
926c33a90ecSJames Feist                 messages::propertyValueTypeError(response->res, "Direction",
927c33a90ecSJames Feist                                                  *direction);
928c33a90ecSJames Feist                 return CreatePIDRet::fail;
929c33a90ecSJames Feist             }
930c33a90ecSJames Feist             output["Class"] = *direction;
931c33a90ecSJames Feist         }
93283ff9ab6SJames Feist     }
93383ff9ab6SJames Feist     else
93483ff9ab6SJames Feist     {
9355f2caaefSJames Feist         BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Type " << type;
93635a62c7cSJason M. Bills         messages::propertyUnknown(response->res, type);
93783ff9ab6SJames Feist         return CreatePIDRet::fail;
93883ff9ab6SJames Feist     }
93983ff9ab6SJames Feist     return CreatePIDRet::patch;
94083ff9ab6SJames Feist }
941*73df0db0SJames Feist struct GetPIDValues : std::enable_shared_from_this<GetPIDValues>
942*73df0db0SJames Feist {
94383ff9ab6SJames Feist 
944*73df0db0SJames Feist     GetPIDValues(const std::shared_ptr<AsyncResp>& asyncResp) :
945*73df0db0SJames Feist         asyncResp(asyncResp)
946*73df0db0SJames Feist 
9471abe55efSEd Tanous     {
9489c310685SBorawski.Lukasz     }
9499c310685SBorawski.Lukasz 
950*73df0db0SJames Feist     void run()
9515b4aa86bSJames Feist     {
952*73df0db0SJames Feist         std::shared_ptr<GetPIDValues> self = shared_from_this();
953*73df0db0SJames Feist 
954*73df0db0SJames Feist         // get all configurations
9555b4aa86bSJames Feist         crow::connections::systemBus->async_method_call(
956*73df0db0SJames Feist             [self](const boost::system::error_code ec,
9575b4aa86bSJames Feist                    const crow::openbmc_mapper::GetSubTreeType& subtree) {
9585b4aa86bSJames Feist                 if (ec)
9595b4aa86bSJames Feist                 {
9605b4aa86bSJames Feist                     BMCWEB_LOG_ERROR << ec;
961*73df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
962*73df0db0SJames Feist                     return;
963*73df0db0SJames Feist                 }
964*73df0db0SJames Feist                 self->subtree = subtree;
965*73df0db0SJames Feist             },
966*73df0db0SJames Feist             "xyz.openbmc_project.ObjectMapper",
967*73df0db0SJames Feist             "/xyz/openbmc_project/object_mapper",
968*73df0db0SJames Feist             "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0,
969*73df0db0SJames Feist             std::array<const char*, 4>{
970*73df0db0SJames Feist                 pidConfigurationIface, pidZoneConfigurationIface,
971*73df0db0SJames Feist                 objectManagerIface, stepwiseConfigurationIface});
972*73df0db0SJames Feist 
973*73df0db0SJames Feist         // at the same time get the selected profile
974*73df0db0SJames Feist         crow::connections::systemBus->async_method_call(
975*73df0db0SJames Feist             [self](const boost::system::error_code ec,
976*73df0db0SJames Feist                    const crow::openbmc_mapper::GetSubTreeType& subtree) {
977*73df0db0SJames Feist                 if (ec || subtree.empty())
978*73df0db0SJames Feist                 {
979*73df0db0SJames Feist                     return;
980*73df0db0SJames Feist                 }
981*73df0db0SJames Feist                 if (subtree[0].second.size() != 1)
982*73df0db0SJames Feist                 {
983*73df0db0SJames Feist                     // invalid mapper response, should never happen
984*73df0db0SJames Feist                     BMCWEB_LOG_ERROR << "GetPIDValues: Mapper Error";
985*73df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
9865b4aa86bSJames Feist                     return;
9875b4aa86bSJames Feist                 }
9885b4aa86bSJames Feist 
989*73df0db0SJames Feist                 const std::string& path = subtree[0].first;
990*73df0db0SJames Feist                 const std::string& owner = subtree[0].second[0].first;
991*73df0db0SJames Feist                 crow::connections::systemBus->async_method_call(
992*73df0db0SJames Feist                     [path, owner, self](
993*73df0db0SJames Feist                         const boost::system::error_code ec,
994*73df0db0SJames Feist                         const boost::container::flat_map<
995*73df0db0SJames Feist                             std::string, std::variant<std::vector<std::string>,
996*73df0db0SJames Feist                                                       std::string>>& resp) {
997*73df0db0SJames Feist                         if (ec)
998*73df0db0SJames Feist                         {
999*73df0db0SJames Feist                             BMCWEB_LOG_ERROR << "GetPIDValues: Can't get "
1000*73df0db0SJames Feist                                                 "thermalModeIface "
1001*73df0db0SJames Feist                                              << path;
1002*73df0db0SJames Feist                             messages::internalError(self->asyncResp->res);
1003*73df0db0SJames Feist                             return;
1004*73df0db0SJames Feist                         }
1005*73df0db0SJames Feist                         const std::string* current;
1006*73df0db0SJames Feist                         const std::vector<std::string>* supported;
1007*73df0db0SJames Feist                         for (auto& [key, value] : resp)
1008*73df0db0SJames Feist                         {
1009*73df0db0SJames Feist                             if (key == "Current")
1010*73df0db0SJames Feist                             {
1011*73df0db0SJames Feist                                 current = std::get_if<std::string>(&value);
1012*73df0db0SJames Feist                                 if (current == nullptr)
1013*73df0db0SJames Feist                                 {
1014*73df0db0SJames Feist                                     BMCWEB_LOG_ERROR
1015*73df0db0SJames Feist                                         << "GetPIDValues: thermal mode "
1016*73df0db0SJames Feist                                            "iface invalid "
1017*73df0db0SJames Feist                                         << path;
1018*73df0db0SJames Feist                                     messages::internalError(
1019*73df0db0SJames Feist                                         self->asyncResp->res);
1020*73df0db0SJames Feist                                     return;
1021*73df0db0SJames Feist                                 }
1022*73df0db0SJames Feist                             }
1023*73df0db0SJames Feist                             if (key == "Supported")
1024*73df0db0SJames Feist                             {
1025*73df0db0SJames Feist                                 supported =
1026*73df0db0SJames Feist                                     std::get_if<std::vector<std::string>>(
1027*73df0db0SJames Feist                                         &value);
1028*73df0db0SJames Feist                                 if (supported == nullptr)
1029*73df0db0SJames Feist                                 {
1030*73df0db0SJames Feist                                     BMCWEB_LOG_ERROR
1031*73df0db0SJames Feist                                         << "GetPIDValues: thermal mode "
1032*73df0db0SJames Feist                                            "iface invalid"
1033*73df0db0SJames Feist                                         << path;
1034*73df0db0SJames Feist                                     messages::internalError(
1035*73df0db0SJames Feist                                         self->asyncResp->res);
1036*73df0db0SJames Feist                                     return;
1037*73df0db0SJames Feist                                 }
1038*73df0db0SJames Feist                             }
1039*73df0db0SJames Feist                         }
1040*73df0db0SJames Feist                         if (current == nullptr || supported == nullptr)
1041*73df0db0SJames Feist                         {
1042*73df0db0SJames Feist                             BMCWEB_LOG_ERROR << "GetPIDValues: thermal mode "
1043*73df0db0SJames Feist                                                 "iface invalid "
1044*73df0db0SJames Feist                                              << path;
1045*73df0db0SJames Feist                             messages::internalError(self->asyncResp->res);
1046*73df0db0SJames Feist                             return;
1047*73df0db0SJames Feist                         }
1048*73df0db0SJames Feist                         self->currentProfile = *current;
1049*73df0db0SJames Feist                         self->supportedProfiles = *supported;
1050*73df0db0SJames Feist                     },
1051*73df0db0SJames Feist                     owner, path, "org.freedesktop.DBus.Properties", "GetAll",
1052*73df0db0SJames Feist                     thermalModeIface);
1053*73df0db0SJames Feist             },
1054*73df0db0SJames Feist             "xyz.openbmc_project.ObjectMapper",
1055*73df0db0SJames Feist             "/xyz/openbmc_project/object_mapper",
1056*73df0db0SJames Feist             "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0,
1057*73df0db0SJames Feist             std::array<const char*, 1>{thermalModeIface});
1058*73df0db0SJames Feist     }
1059*73df0db0SJames Feist 
1060*73df0db0SJames Feist     ~GetPIDValues()
1061*73df0db0SJames Feist     {
1062*73df0db0SJames Feist         if (asyncResp->res.result() != boost::beast::http::status::ok)
1063*73df0db0SJames Feist         {
1064*73df0db0SJames Feist             return;
1065*73df0db0SJames Feist         }
10665b4aa86bSJames Feist         // create map of <connection, path to objMgr>>
1067*73df0db0SJames Feist         boost::container::flat_map<std::string, std::string> objectMgrPaths;
10686bce33bcSJames Feist         boost::container::flat_set<std::string> calledConnections;
10695b4aa86bSJames Feist         for (const auto& pathGroup : subtree)
10705b4aa86bSJames Feist         {
10715b4aa86bSJames Feist             for (const auto& connectionGroup : pathGroup.second)
10725b4aa86bSJames Feist             {
10736bce33bcSJames Feist                 auto findConnection =
10746bce33bcSJames Feist                     calledConnections.find(connectionGroup.first);
10756bce33bcSJames Feist                 if (findConnection != calledConnections.end())
10766bce33bcSJames Feist                 {
10776bce33bcSJames Feist                     break;
10786bce33bcSJames Feist                 }
1079*73df0db0SJames Feist                 for (const std::string& interface : connectionGroup.second)
10805b4aa86bSJames Feist                 {
10815b4aa86bSJames Feist                     if (interface == objectManagerIface)
10825b4aa86bSJames Feist                     {
1083*73df0db0SJames Feist                         objectMgrPaths[connectionGroup.first] = pathGroup.first;
10845b4aa86bSJames Feist                     }
10855b4aa86bSJames Feist                     // this list is alphabetical, so we
10865b4aa86bSJames Feist                     // should have found the objMgr by now
10875b4aa86bSJames Feist                     if (interface == pidConfigurationIface ||
1088b7a08d04SJames Feist                         interface == pidZoneConfigurationIface ||
1089b7a08d04SJames Feist                         interface == stepwiseConfigurationIface)
10905b4aa86bSJames Feist                     {
10915b4aa86bSJames Feist                         auto findObjMgr =
10925b4aa86bSJames Feist                             objectMgrPaths.find(connectionGroup.first);
10935b4aa86bSJames Feist                         if (findObjMgr == objectMgrPaths.end())
10945b4aa86bSJames Feist                         {
10955b4aa86bSJames Feist                             BMCWEB_LOG_DEBUG << connectionGroup.first
10965b4aa86bSJames Feist                                              << "Has no Object Manager";
10975b4aa86bSJames Feist                             continue;
10985b4aa86bSJames Feist                         }
10996bce33bcSJames Feist 
11006bce33bcSJames Feist                         calledConnections.insert(connectionGroup.first);
11016bce33bcSJames Feist 
1102*73df0db0SJames Feist                         asyncPopulatePid(findObjMgr->first, findObjMgr->second,
1103*73df0db0SJames Feist                                          currentProfile, supportedProfiles,
1104*73df0db0SJames Feist                                          asyncResp);
11055b4aa86bSJames Feist                         break;
11065b4aa86bSJames Feist                     }
11075b4aa86bSJames Feist                 }
11085b4aa86bSJames Feist             }
11095b4aa86bSJames Feist         }
1110*73df0db0SJames Feist     }
1111*73df0db0SJames Feist 
1112*73df0db0SJames Feist     std::vector<std::string> supportedProfiles;
1113*73df0db0SJames Feist     std::string currentProfile;
1114*73df0db0SJames Feist     crow::openbmc_mapper::GetSubTreeType subtree;
1115*73df0db0SJames Feist     std::shared_ptr<AsyncResp> asyncResp;
1116*73df0db0SJames Feist };
1117*73df0db0SJames Feist 
1118*73df0db0SJames Feist struct SetPIDValues : std::enable_shared_from_this<SetPIDValues>
1119*73df0db0SJames Feist {
1120*73df0db0SJames Feist 
1121*73df0db0SJames Feist     SetPIDValues(const std::shared_ptr<AsyncResp>& asyncResp,
1122*73df0db0SJames Feist                  nlohmann::json& data) :
1123*73df0db0SJames Feist         asyncResp(asyncResp)
1124*73df0db0SJames Feist     {
1125*73df0db0SJames Feist 
1126*73df0db0SJames Feist         std::optional<nlohmann::json> pidControllers;
1127*73df0db0SJames Feist         std::optional<nlohmann::json> fanControllers;
1128*73df0db0SJames Feist         std::optional<nlohmann::json> fanZones;
1129*73df0db0SJames Feist         std::optional<nlohmann::json> stepwiseControllers;
1130*73df0db0SJames Feist 
1131*73df0db0SJames Feist         if (!redfish::json_util::readJson(
1132*73df0db0SJames Feist                 data, asyncResp->res, "PidControllers", pidControllers,
1133*73df0db0SJames Feist                 "FanControllers", fanControllers, "FanZones", fanZones,
1134*73df0db0SJames Feist                 "StepwiseControllers", stepwiseControllers, "Profile", profile))
1135*73df0db0SJames Feist         {
1136*73df0db0SJames Feist             BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property "
1137*73df0db0SJames Feist                              << data.dump();
1138*73df0db0SJames Feist             return;
1139*73df0db0SJames Feist         }
1140*73df0db0SJames Feist         configuration.emplace_back("PidControllers", std::move(pidControllers));
1141*73df0db0SJames Feist         configuration.emplace_back("FanControllers", std::move(fanControllers));
1142*73df0db0SJames Feist         configuration.emplace_back("FanZones", std::move(fanZones));
1143*73df0db0SJames Feist         configuration.emplace_back("StepwiseControllers",
1144*73df0db0SJames Feist                                    std::move(stepwiseControllers));
1145*73df0db0SJames Feist     }
1146*73df0db0SJames Feist     void run()
1147*73df0db0SJames Feist     {
1148*73df0db0SJames Feist         if (asyncResp->res.result() != boost::beast::http::status::ok)
1149*73df0db0SJames Feist         {
1150*73df0db0SJames Feist             return;
1151*73df0db0SJames Feist         }
1152*73df0db0SJames Feist 
1153*73df0db0SJames Feist         std::shared_ptr<SetPIDValues> self = shared_from_this();
1154*73df0db0SJames Feist 
1155*73df0db0SJames Feist         // todo(james): might make sense to do a mapper call here if this
1156*73df0db0SJames Feist         // interface gets more traction
1157*73df0db0SJames Feist         crow::connections::systemBus->async_method_call(
1158*73df0db0SJames Feist             [self](const boost::system::error_code ec,
1159*73df0db0SJames Feist                    dbus::utility::ManagedObjectType& managedObj) {
1160*73df0db0SJames Feist                 if (ec)
1161*73df0db0SJames Feist                 {
1162*73df0db0SJames Feist                     BMCWEB_LOG_ERROR << "Error communicating to Entity Manager";
1163*73df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
1164*73df0db0SJames Feist                     return;
1165*73df0db0SJames Feist                 }
1166*73df0db0SJames Feist                 self->managedObj = std::move(managedObj);
1167*73df0db0SJames Feist             },
1168*73df0db0SJames Feist             "xyz.openbmc_project.EntityManager", "/", objectManagerIface,
1169*73df0db0SJames Feist             "GetManagedObjects");
1170*73df0db0SJames Feist 
1171*73df0db0SJames Feist         // at the same time get the profile information
1172*73df0db0SJames Feist         crow::connections::systemBus->async_method_call(
1173*73df0db0SJames Feist             [self](const boost::system::error_code ec,
1174*73df0db0SJames Feist                    const crow::openbmc_mapper::GetSubTreeType& subtree) {
1175*73df0db0SJames Feist                 if (ec || subtree.empty())
1176*73df0db0SJames Feist                 {
1177*73df0db0SJames Feist                     return;
1178*73df0db0SJames Feist                 }
1179*73df0db0SJames Feist                 if (subtree[0].second.empty())
1180*73df0db0SJames Feist                 {
1181*73df0db0SJames Feist                     // invalid mapper response, should never happen
1182*73df0db0SJames Feist                     BMCWEB_LOG_ERROR << "SetPIDValues: Mapper Error";
1183*73df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
1184*73df0db0SJames Feist                     return;
1185*73df0db0SJames Feist                 }
1186*73df0db0SJames Feist 
1187*73df0db0SJames Feist                 const std::string& path = subtree[0].first;
1188*73df0db0SJames Feist                 const std::string& owner = subtree[0].second[0].first;
1189*73df0db0SJames Feist                 crow::connections::systemBus->async_method_call(
1190*73df0db0SJames Feist                     [self, path, owner](
1191*73df0db0SJames Feist                         const boost::system::error_code ec,
1192*73df0db0SJames Feist                         const boost::container::flat_map<
1193*73df0db0SJames Feist                             std::string, std::variant<std::vector<std::string>,
1194*73df0db0SJames Feist                                                       std::string>>& resp) {
1195*73df0db0SJames Feist                         if (ec)
1196*73df0db0SJames Feist                         {
1197*73df0db0SJames Feist                             BMCWEB_LOG_ERROR << "SetPIDValues: Can't get "
1198*73df0db0SJames Feist                                                 "thermalModeIface "
1199*73df0db0SJames Feist                                              << path;
1200*73df0db0SJames Feist                             messages::internalError(self->asyncResp->res);
1201*73df0db0SJames Feist                             return;
1202*73df0db0SJames Feist                         }
1203*73df0db0SJames Feist                         const std::string* current;
1204*73df0db0SJames Feist                         const std::vector<std::string>* supported;
1205*73df0db0SJames Feist                         for (auto& [key, value] : resp)
1206*73df0db0SJames Feist                         {
1207*73df0db0SJames Feist                             if (key == "Current")
1208*73df0db0SJames Feist                             {
1209*73df0db0SJames Feist                                 current = std::get_if<std::string>(&value);
1210*73df0db0SJames Feist                                 if (current == nullptr)
1211*73df0db0SJames Feist                                 {
1212*73df0db0SJames Feist                                     BMCWEB_LOG_ERROR
1213*73df0db0SJames Feist                                         << "SetPIDValues: thermal mode "
1214*73df0db0SJames Feist                                            "iface invalid "
1215*73df0db0SJames Feist                                         << path;
1216*73df0db0SJames Feist                                     messages::internalError(
1217*73df0db0SJames Feist                                         self->asyncResp->res);
1218*73df0db0SJames Feist                                     return;
1219*73df0db0SJames Feist                                 }
1220*73df0db0SJames Feist                             }
1221*73df0db0SJames Feist                             if (key == "Supported")
1222*73df0db0SJames Feist                             {
1223*73df0db0SJames Feist                                 supported =
1224*73df0db0SJames Feist                                     std::get_if<std::vector<std::string>>(
1225*73df0db0SJames Feist                                         &value);
1226*73df0db0SJames Feist                                 if (supported == nullptr)
1227*73df0db0SJames Feist                                 {
1228*73df0db0SJames Feist                                     BMCWEB_LOG_ERROR
1229*73df0db0SJames Feist                                         << "SetPIDValues: thermal mode "
1230*73df0db0SJames Feist                                            "iface invalid"
1231*73df0db0SJames Feist                                         << path;
1232*73df0db0SJames Feist                                     messages::internalError(
1233*73df0db0SJames Feist                                         self->asyncResp->res);
1234*73df0db0SJames Feist                                     return;
1235*73df0db0SJames Feist                                 }
1236*73df0db0SJames Feist                             }
1237*73df0db0SJames Feist                         }
1238*73df0db0SJames Feist                         if (current == nullptr || supported == nullptr)
1239*73df0db0SJames Feist                         {
1240*73df0db0SJames Feist                             BMCWEB_LOG_ERROR << "SetPIDValues: thermal mode "
1241*73df0db0SJames Feist                                                 "iface invalid "
1242*73df0db0SJames Feist                                              << path;
1243*73df0db0SJames Feist                             messages::internalError(self->asyncResp->res);
1244*73df0db0SJames Feist                             return;
1245*73df0db0SJames Feist                         }
1246*73df0db0SJames Feist                         self->currentProfile = *current;
1247*73df0db0SJames Feist                         self->supportedProfiles = *supported;
1248*73df0db0SJames Feist                         self->profileConnection = owner;
1249*73df0db0SJames Feist                         self->profilePath = path;
1250*73df0db0SJames Feist                     },
1251*73df0db0SJames Feist                     owner, path, "org.freedesktop.DBus.Properties", "GetAll",
1252*73df0db0SJames Feist                     thermalModeIface);
12535b4aa86bSJames Feist             },
12545b4aa86bSJames Feist             "xyz.openbmc_project.ObjectMapper",
12555b4aa86bSJames Feist             "/xyz/openbmc_project/object_mapper",
12565b4aa86bSJames Feist             "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0,
1257*73df0db0SJames Feist             std::array<const char*, 1>{thermalModeIface});
1258*73df0db0SJames Feist     }
1259*73df0db0SJames Feist     ~SetPIDValues()
1260*73df0db0SJames Feist     {
1261*73df0db0SJames Feist         if (asyncResp->res.result() != boost::beast::http::status::ok)
1262*73df0db0SJames Feist         {
1263*73df0db0SJames Feist             return;
12645b4aa86bSJames Feist         }
12655b4aa86bSJames Feist 
1266*73df0db0SJames Feist         std::shared_ptr<AsyncResp> response = asyncResp;
1267*73df0db0SJames Feist 
1268*73df0db0SJames Feist         if (profile)
1269*73df0db0SJames Feist         {
1270*73df0db0SJames Feist             if (std::find(supportedProfiles.begin(), supportedProfiles.end(),
1271*73df0db0SJames Feist                           *profile) == supportedProfiles.end())
1272*73df0db0SJames Feist             {
1273*73df0db0SJames Feist                 messages::actionParameterUnknown(response->res, "Profile",
1274*73df0db0SJames Feist                                                  *profile);
1275*73df0db0SJames Feist                 return;
1276*73df0db0SJames Feist             }
1277*73df0db0SJames Feist             currentProfile = *profile;
1278*73df0db0SJames Feist             crow::connections::systemBus->async_method_call(
1279*73df0db0SJames Feist                 [response](const boost::system::error_code ec) {
1280*73df0db0SJames Feist                     if (ec)
1281*73df0db0SJames Feist                     {
1282*73df0db0SJames Feist                         BMCWEB_LOG_ERROR << "Error patching profile" << ec;
1283*73df0db0SJames Feist                         messages::internalError(response->res);
1284*73df0db0SJames Feist                     }
1285*73df0db0SJames Feist                 },
1286*73df0db0SJames Feist                 profileConnection, profilePath,
1287*73df0db0SJames Feist                 "org.freedesktop.DBus.Properties", "Set", thermalModeIface,
1288*73df0db0SJames Feist                 "Current", std::variant<std::string>(*profile));
1289*73df0db0SJames Feist         }
1290*73df0db0SJames Feist 
1291*73df0db0SJames Feist         for (auto& containerPair : configuration)
1292*73df0db0SJames Feist         {
1293*73df0db0SJames Feist             auto& container = containerPair.second;
1294*73df0db0SJames Feist             if (!container)
1295*73df0db0SJames Feist             {
1296*73df0db0SJames Feist                 continue;
1297*73df0db0SJames Feist             }
1298*73df0db0SJames Feist             std::string& type = containerPair.first;
1299*73df0db0SJames Feist 
1300*73df0db0SJames Feist             for (nlohmann::json::iterator it = container->begin();
1301*73df0db0SJames Feist                  it != container->end(); it++)
1302*73df0db0SJames Feist             {
1303*73df0db0SJames Feist                 const auto& name = it.key();
1304*73df0db0SJames Feist                 auto pathItr =
1305*73df0db0SJames Feist                     std::find_if(managedObj.begin(), managedObj.end(),
1306*73df0db0SJames Feist                                  [&name](const auto& obj) {
1307*73df0db0SJames Feist                                      return boost::algorithm::ends_with(
1308*73df0db0SJames Feist                                          obj.first.str, "/" + name);
1309*73df0db0SJames Feist                                  });
1310*73df0db0SJames Feist                 boost::container::flat_map<std::string,
1311*73df0db0SJames Feist                                            dbus::utility::DbusVariantType>
1312*73df0db0SJames Feist                     output;
1313*73df0db0SJames Feist 
1314*73df0db0SJames Feist                 output.reserve(16); // The pid interface length
1315*73df0db0SJames Feist 
1316*73df0db0SJames Feist                 // determines if we're patching entity-manager or
1317*73df0db0SJames Feist                 // creating a new object
1318*73df0db0SJames Feist                 bool createNewObject = (pathItr == managedObj.end());
1319*73df0db0SJames Feist                 std::string iface;
1320*73df0db0SJames Feist                 if (type == "PidControllers" || type == "FanControllers")
1321*73df0db0SJames Feist                 {
1322*73df0db0SJames Feist                     iface = pidConfigurationIface;
1323*73df0db0SJames Feist                     if (!createNewObject &&
1324*73df0db0SJames Feist                         pathItr->second.find(pidConfigurationIface) ==
1325*73df0db0SJames Feist                             pathItr->second.end())
1326*73df0db0SJames Feist                     {
1327*73df0db0SJames Feist                         createNewObject = true;
1328*73df0db0SJames Feist                     }
1329*73df0db0SJames Feist                 }
1330*73df0db0SJames Feist                 else if (type == "FanZones")
1331*73df0db0SJames Feist                 {
1332*73df0db0SJames Feist                     iface = pidZoneConfigurationIface;
1333*73df0db0SJames Feist                     if (!createNewObject &&
1334*73df0db0SJames Feist                         pathItr->second.find(pidZoneConfigurationIface) ==
1335*73df0db0SJames Feist                             pathItr->second.end())
1336*73df0db0SJames Feist                     {
1337*73df0db0SJames Feist 
1338*73df0db0SJames Feist                         createNewObject = true;
1339*73df0db0SJames Feist                     }
1340*73df0db0SJames Feist                 }
1341*73df0db0SJames Feist                 else if (type == "StepwiseControllers")
1342*73df0db0SJames Feist                 {
1343*73df0db0SJames Feist                     iface = stepwiseConfigurationIface;
1344*73df0db0SJames Feist                     if (!createNewObject &&
1345*73df0db0SJames Feist                         pathItr->second.find(stepwiseConfigurationIface) ==
1346*73df0db0SJames Feist                             pathItr->second.end())
1347*73df0db0SJames Feist                     {
1348*73df0db0SJames Feist                         createNewObject = true;
1349*73df0db0SJames Feist                     }
1350*73df0db0SJames Feist                 }
1351*73df0db0SJames Feist                 BMCWEB_LOG_DEBUG << "Create new = " << createNewObject << "\n";
1352*73df0db0SJames Feist                 output["Name"] = boost::replace_all_copy(name, "_", " ");
1353*73df0db0SJames Feist 
1354*73df0db0SJames Feist                 std::string chassis;
1355*73df0db0SJames Feist                 CreatePIDRet ret = createPidInterface(
1356*73df0db0SJames Feist                     response, type, it, pathItr->first.str, managedObj,
1357*73df0db0SJames Feist                     createNewObject, output, chassis, currentProfile);
1358*73df0db0SJames Feist                 if (ret == CreatePIDRet::fail)
1359*73df0db0SJames Feist                 {
1360*73df0db0SJames Feist                     return;
1361*73df0db0SJames Feist                 }
1362*73df0db0SJames Feist                 else if (ret == CreatePIDRet::del)
1363*73df0db0SJames Feist                 {
1364*73df0db0SJames Feist                     continue;
1365*73df0db0SJames Feist                 }
1366*73df0db0SJames Feist 
1367*73df0db0SJames Feist                 if (!createNewObject)
1368*73df0db0SJames Feist                 {
1369*73df0db0SJames Feist                     for (const auto& property : output)
1370*73df0db0SJames Feist                     {
1371*73df0db0SJames Feist                         crow::connections::systemBus->async_method_call(
1372*73df0db0SJames Feist                             [response,
1373*73df0db0SJames Feist                              propertyName{std::string(property.first)}](
1374*73df0db0SJames Feist                                 const boost::system::error_code ec) {
1375*73df0db0SJames Feist                                 if (ec)
1376*73df0db0SJames Feist                                 {
1377*73df0db0SJames Feist                                     BMCWEB_LOG_ERROR << "Error patching "
1378*73df0db0SJames Feist                                                      << propertyName << ": "
1379*73df0db0SJames Feist                                                      << ec;
1380*73df0db0SJames Feist                                     messages::internalError(response->res);
1381*73df0db0SJames Feist                                     return;
1382*73df0db0SJames Feist                                 }
1383*73df0db0SJames Feist                                 messages::success(response->res);
1384*73df0db0SJames Feist                             },
1385*73df0db0SJames Feist                             "xyz.openbmc_project.EntityManager",
1386*73df0db0SJames Feist                             pathItr->first.str,
1387*73df0db0SJames Feist                             "org.freedesktop.DBus.Properties", "Set", iface,
1388*73df0db0SJames Feist                             property.first, property.second);
1389*73df0db0SJames Feist                     }
1390*73df0db0SJames Feist                 }
1391*73df0db0SJames Feist                 else
1392*73df0db0SJames Feist                 {
1393*73df0db0SJames Feist                     if (chassis.empty())
1394*73df0db0SJames Feist                     {
1395*73df0db0SJames Feist                         BMCWEB_LOG_ERROR << "Failed to get chassis from config";
1396*73df0db0SJames Feist                         messages::invalidObject(response->res, name);
1397*73df0db0SJames Feist                         return;
1398*73df0db0SJames Feist                     }
1399*73df0db0SJames Feist 
1400*73df0db0SJames Feist                     bool foundChassis = false;
1401*73df0db0SJames Feist                     for (const auto& obj : managedObj)
1402*73df0db0SJames Feist                     {
1403*73df0db0SJames Feist                         if (boost::algorithm::ends_with(obj.first.str, chassis))
1404*73df0db0SJames Feist                         {
1405*73df0db0SJames Feist                             chassis = obj.first.str;
1406*73df0db0SJames Feist                             foundChassis = true;
1407*73df0db0SJames Feist                             break;
1408*73df0db0SJames Feist                         }
1409*73df0db0SJames Feist                     }
1410*73df0db0SJames Feist                     if (!foundChassis)
1411*73df0db0SJames Feist                     {
1412*73df0db0SJames Feist                         BMCWEB_LOG_ERROR << "Failed to find chassis on dbus";
1413*73df0db0SJames Feist                         messages::resourceMissingAtURI(
1414*73df0db0SJames Feist                             response->res, "/redfish/v1/Chassis/" + chassis);
1415*73df0db0SJames Feist                         return;
1416*73df0db0SJames Feist                     }
1417*73df0db0SJames Feist 
1418*73df0db0SJames Feist                     crow::connections::systemBus->async_method_call(
1419*73df0db0SJames Feist                         [response](const boost::system::error_code ec) {
1420*73df0db0SJames Feist                             if (ec)
1421*73df0db0SJames Feist                             {
1422*73df0db0SJames Feist                                 BMCWEB_LOG_ERROR << "Error Adding Pid Object "
1423*73df0db0SJames Feist                                                  << ec;
1424*73df0db0SJames Feist                                 messages::internalError(response->res);
1425*73df0db0SJames Feist                                 return;
1426*73df0db0SJames Feist                             }
1427*73df0db0SJames Feist                             messages::success(response->res);
1428*73df0db0SJames Feist                         },
1429*73df0db0SJames Feist                         "xyz.openbmc_project.EntityManager", chassis,
1430*73df0db0SJames Feist                         "xyz.openbmc_project.AddObject", "AddObject", output);
1431*73df0db0SJames Feist                 }
1432*73df0db0SJames Feist             }
1433*73df0db0SJames Feist         }
1434*73df0db0SJames Feist     }
1435*73df0db0SJames Feist     std::shared_ptr<AsyncResp> asyncResp;
1436*73df0db0SJames Feist     std::vector<std::pair<std::string, std::optional<nlohmann::json>>>
1437*73df0db0SJames Feist         configuration;
1438*73df0db0SJames Feist     std::optional<std::string> profile;
1439*73df0db0SJames Feist     dbus::utility::ManagedObjectType managedObj;
1440*73df0db0SJames Feist     std::vector<std::string> supportedProfiles;
1441*73df0db0SJames Feist     std::string currentProfile;
1442*73df0db0SJames Feist     std::string profileConnection;
1443*73df0db0SJames Feist     std::string profilePath;
1444*73df0db0SJames Feist };
1445*73df0db0SJames Feist 
1446*73df0db0SJames Feist class Manager : public Node
1447*73df0db0SJames Feist {
1448*73df0db0SJames Feist   public:
1449*73df0db0SJames Feist     Manager(CrowApp& app) : Node(app, "/redfish/v1/Managers/bmc/")
1450*73df0db0SJames Feist     {
1451*73df0db0SJames Feist         uuid = app.template getMiddleware<crow::persistent_data::Middleware>()
1452*73df0db0SJames Feist                    .systemUuid;
1453*73df0db0SJames Feist         entityPrivileges = {
1454*73df0db0SJames Feist             {boost::beast::http::verb::get, {{"Login"}}},
1455*73df0db0SJames Feist             {boost::beast::http::verb::head, {{"Login"}}},
1456*73df0db0SJames Feist             {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
1457*73df0db0SJames Feist             {boost::beast::http::verb::put, {{"ConfigureManager"}}},
1458*73df0db0SJames Feist             {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
1459*73df0db0SJames Feist             {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
1460*73df0db0SJames Feist     }
1461*73df0db0SJames Feist 
1462*73df0db0SJames Feist   private:
146355c7b7a2SEd Tanous     void doGet(crow::Response& res, const crow::Request& req,
14641abe55efSEd Tanous                const std::vector<std::string>& params) override
14651abe55efSEd Tanous     {
14660f74e643SEd Tanous         res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc";
14670f74e643SEd Tanous         res.jsonValue["@odata.type"] = "#Manager.v1_3_0.Manager";
14680f74e643SEd Tanous         res.jsonValue["@odata.context"] =
14690f74e643SEd Tanous             "/redfish/v1/$metadata#Manager.Manager";
14700f74e643SEd Tanous         res.jsonValue["Id"] = "bmc";
14710f74e643SEd Tanous         res.jsonValue["Name"] = "OpenBmc Manager";
14720f74e643SEd Tanous         res.jsonValue["Description"] = "Baseboard Management Controller";
14730f74e643SEd Tanous         res.jsonValue["PowerState"] = "On";
1474029573d4SEd Tanous         res.jsonValue["Status"] = {{"State", "Enabled"}, {"Health", "OK"}};
14750f74e643SEd Tanous         res.jsonValue["ManagerType"] = "BMC";
14763602e232SEd Tanous         res.jsonValue["UUID"] = systemd_utils::getUuid();
14773602e232SEd Tanous         res.jsonValue["ServiceEntryPointUUID"] = uuid;
14780f74e643SEd Tanous         res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model
14790f74e643SEd Tanous 
14800f74e643SEd Tanous         res.jsonValue["LogServices"] = {
14810f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/LogServices"}};
14820f74e643SEd Tanous 
14830f74e643SEd Tanous         res.jsonValue["NetworkProtocol"] = {
14840f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/NetworkProtocol"}};
14850f74e643SEd Tanous 
14860f74e643SEd Tanous         res.jsonValue["EthernetInterfaces"] = {
14870f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/EthernetInterfaces"}};
14880f74e643SEd Tanous         // default oem data
14890f74e643SEd Tanous         nlohmann::json& oem = res.jsonValue["Oem"];
14900f74e643SEd Tanous         nlohmann::json& oemOpenbmc = oem["OpenBmc"];
14910f74e643SEd Tanous         oem["@odata.type"] = "#OemManager.Oem";
14920f74e643SEd Tanous         oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem";
14930f74e643SEd Tanous         oem["@odata.context"] = "/redfish/v1/$metadata#OemManager.Oem";
14940f74e643SEd Tanous         oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc";
14950f74e643SEd Tanous         oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc";
14960f74e643SEd Tanous         oemOpenbmc["@odata.context"] =
14970f74e643SEd Tanous             "/redfish/v1/$metadata#OemManager.OpenBmc";
14980f74e643SEd Tanous 
1499ed5befbdSJennifer Lee         // Update Actions object.
15000f74e643SEd Tanous         nlohmann::json& manager_reset =
15010f74e643SEd Tanous             res.jsonValue["Actions"]["#Manager.Reset"];
1502ed5befbdSJennifer Lee         manager_reset["target"] =
1503ed5befbdSJennifer Lee             "/redfish/v1/Managers/bmc/Actions/Manager.Reset";
1504ed5befbdSJennifer Lee         manager_reset["ResetType@Redfish.AllowableValues"] = {
1505ed5befbdSJennifer Lee             "GracefulRestart"};
1506ca537928SJennifer Lee 
1507cb92c03bSAndrew Geissler         res.jsonValue["DateTime"] = crow::utility::dateTimeNow();
1508474bfad5SSantosh Puranik 
1509474bfad5SSantosh Puranik         // Fill in GraphicalConsole and SerialConsole info
1510474bfad5SSantosh Puranik         res.jsonValue["SerialConsole"]["ServiceEnabled"] = true;
1511474bfad5SSantosh Puranik         res.jsonValue["SerialConsole"]["ConnectTypesSupported"] = {"IPMI",
1512474bfad5SSantosh Puranik                                                                    "SSH"};
1513474bfad5SSantosh Puranik         // TODO (Santosh) : Uncomment when KVM support is in.
1514474bfad5SSantosh Puranik         // res.jsonValue["GraphicalConsole"]["ServiceEnabled"] = true;
1515474bfad5SSantosh Puranik         // res.jsonValue["GraphicalConsole"]["ConnectTypesSupported"] =
1516474bfad5SSantosh Puranik         //    {"KVMIP"};
1517474bfad5SSantosh Puranik 
1518603a6640SGunnar Mills         res.jsonValue["Links"]["ManagerForServers@odata.count"] = 1;
1519603a6640SGunnar Mills         res.jsonValue["Links"]["ManagerForServers"] = {
1520603a6640SGunnar Mills             {{"@odata.id", "/redfish/v1/Systems/system"}}};
152126f03899SShawn McCarney 
1522ed5befbdSJennifer Lee         std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res);
15235b4aa86bSJames Feist 
1524ca537928SJennifer Lee         crow::connections::systemBus->async_method_call(
1525ca537928SJennifer Lee             [asyncResp](const boost::system::error_code ec,
15265b4aa86bSJames Feist                         const dbus::utility::ManagedObjectType& resp) {
1527ca537928SJennifer Lee                 if (ec)
1528ca537928SJennifer Lee                 {
1529ca537928SJennifer Lee                     BMCWEB_LOG_ERROR << "Error while getting Software Version";
1530f12894f8SJason M. Bills                     messages::internalError(asyncResp->res);
1531ca537928SJennifer Lee                     return;
1532ca537928SJennifer Lee                 }
1533ca537928SJennifer Lee 
1534ca537928SJennifer Lee                 for (auto& objpath : resp)
1535ca537928SJennifer Lee                 {
1536ca537928SJennifer Lee                     for (auto& interface : objpath.second)
1537ca537928SJennifer Lee                     {
15385f2caaefSJames Feist                         // If interface is
15395f2caaefSJames Feist                         // xyz.openbmc_project.Software.Version, this is
15405f2caaefSJames Feist                         // what we're looking for.
1541ca537928SJennifer Lee                         if (interface.first ==
1542ca537928SJennifer Lee                             "xyz.openbmc_project.Software.Version")
1543ca537928SJennifer Lee                         {
1544ca537928SJennifer Lee                             // Cut out everyting until last "/", ...
1545ca537928SJennifer Lee                             for (auto& property : interface.second)
1546ca537928SJennifer Lee                             {
1547ca537928SJennifer Lee                                 if (property.first == "Version")
1548ca537928SJennifer Lee                                 {
1549ca537928SJennifer Lee                                     const std::string* value =
1550abf2add6SEd Tanous                                         std::get_if<std::string>(
1551abf2add6SEd Tanous                                             &property.second);
1552ca537928SJennifer Lee                                     if (value == nullptr)
1553ca537928SJennifer Lee                                     {
1554ca537928SJennifer Lee                                         continue;
1555ca537928SJennifer Lee                                     }
1556ca537928SJennifer Lee                                     asyncResp->res
1557ca537928SJennifer Lee                                         .jsonValue["FirmwareVersion"] = *value;
1558ca537928SJennifer Lee                                 }
1559ca537928SJennifer Lee                             }
1560ca537928SJennifer Lee                         }
1561ca537928SJennifer Lee                     }
1562ca537928SJennifer Lee                 }
1563ca537928SJennifer Lee             },
1564ca537928SJennifer Lee             "xyz.openbmc_project.Software.BMC.Updater",
1565ca537928SJennifer Lee             "/xyz/openbmc_project/software",
1566ca537928SJennifer Lee             "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
1567*73df0db0SJames Feist         auto pids = std::make_shared<GetPIDValues>(asyncResp);
1568*73df0db0SJames Feist         pids->run();
156983ff9ab6SJames Feist     }
15705b4aa86bSJames Feist 
15715b4aa86bSJames Feist     void doPatch(crow::Response& res, const crow::Request& req,
15725b4aa86bSJames Feist                  const std::vector<std::string>& params) override
15735b4aa86bSJames Feist     {
15740627a2c7SEd Tanous         std::optional<nlohmann::json> oem;
1575af5d6058SSantosh Puranik         std::optional<std::string> datetime;
15760627a2c7SEd Tanous 
1577af5d6058SSantosh Puranik         if (!json_util::readJson(req, res, "Oem", oem, "DateTime", datetime))
157883ff9ab6SJames Feist         {
157983ff9ab6SJames Feist             return;
158083ff9ab6SJames Feist         }
15810627a2c7SEd Tanous 
158283ff9ab6SJames Feist         std::shared_ptr<AsyncResp> response = std::make_shared<AsyncResp>(res);
15830627a2c7SEd Tanous 
15840627a2c7SEd Tanous         if (oem)
158583ff9ab6SJames Feist         {
15865f2caaefSJames Feist             std::optional<nlohmann::json> openbmc;
158743b761d0SEd Tanous             if (!redfish::json_util::readJson(*oem, res, "OpenBmc", openbmc))
158883ff9ab6SJames Feist             {
158943b761d0SEd Tanous                 BMCWEB_LOG_ERROR << "Line:" << __LINE__ << ", Illegal Property "
159043b761d0SEd Tanous                                  << oem->dump();
159183ff9ab6SJames Feist                 return;
159283ff9ab6SJames Feist             }
15935f2caaefSJames Feist             if (openbmc)
159483ff9ab6SJames Feist             {
15955f2caaefSJames Feist                 std::optional<nlohmann::json> fan;
159643b761d0SEd Tanous                 if (!redfish::json_util::readJson(*openbmc, res, "Fan", fan))
159783ff9ab6SJames Feist                 {
15985f2caaefSJames Feist                     BMCWEB_LOG_ERROR << "Line:" << __LINE__
15995f2caaefSJames Feist                                      << ", Illegal Property "
16005f2caaefSJames Feist                                      << openbmc->dump();
160183ff9ab6SJames Feist                     return;
160283ff9ab6SJames Feist                 }
16035f2caaefSJames Feist                 if (fan)
160483ff9ab6SJames Feist                 {
1605*73df0db0SJames Feist                     auto pid = std::make_shared<SetPIDValues>(response, *fan);
1606*73df0db0SJames Feist                     pid->run();
160783ff9ab6SJames Feist                 }
160883ff9ab6SJames Feist             }
160983ff9ab6SJames Feist         }
1610af5d6058SSantosh Puranik         if (datetime)
1611af5d6058SSantosh Puranik         {
1612af5d6058SSantosh Puranik             setDateTime(response, std::move(*datetime));
1613af5d6058SSantosh Puranik         }
1614af5d6058SSantosh Puranik     }
1615af5d6058SSantosh Puranik 
1616af5d6058SSantosh Puranik     void setDateTime(std::shared_ptr<AsyncResp> aResp,
1617af5d6058SSantosh Puranik                      std::string datetime) const
1618af5d6058SSantosh Puranik     {
1619af5d6058SSantosh Puranik         BMCWEB_LOG_DEBUG << "Set date time: " << datetime;
1620af5d6058SSantosh Puranik 
1621af5d6058SSantosh Puranik         std::stringstream stream(datetime);
1622af5d6058SSantosh Puranik         // Convert from ISO 8601 to boost local_time
1623af5d6058SSantosh Puranik         // (BMC only has time in UTC)
1624af5d6058SSantosh Puranik         boost::posix_time::ptime posixTime;
1625af5d6058SSantosh Puranik         boost::posix_time::ptime epoch(boost::gregorian::date(1970, 1, 1));
1626af5d6058SSantosh Puranik         // Facet gets deleted with the stringsteam
1627af5d6058SSantosh Puranik         auto ifc = std::make_unique<boost::local_time::local_time_input_facet>(
1628af5d6058SSantosh Puranik             "%Y-%m-%d %H:%M:%S%F %ZP");
1629af5d6058SSantosh Puranik         stream.imbue(std::locale(stream.getloc(), ifc.release()));
1630af5d6058SSantosh Puranik 
1631af5d6058SSantosh Puranik         boost::local_time::local_date_time ldt(
1632af5d6058SSantosh Puranik             boost::local_time::not_a_date_time);
1633af5d6058SSantosh Puranik 
1634af5d6058SSantosh Puranik         if (stream >> ldt)
1635af5d6058SSantosh Puranik         {
1636af5d6058SSantosh Puranik             posixTime = ldt.utc_time();
1637af5d6058SSantosh Puranik             boost::posix_time::time_duration dur = posixTime - epoch;
1638af5d6058SSantosh Puranik             uint64_t durMicroSecs =
1639af5d6058SSantosh Puranik                 static_cast<uint64_t>(dur.total_microseconds());
1640af5d6058SSantosh Puranik             crow::connections::systemBus->async_method_call(
1641af5d6058SSantosh Puranik                 [aResp{std::move(aResp)}, datetime{std::move(datetime)}](
1642af5d6058SSantosh Puranik                     const boost::system::error_code ec) {
1643af5d6058SSantosh Puranik                     if (ec)
1644af5d6058SSantosh Puranik                     {
1645af5d6058SSantosh Puranik                         BMCWEB_LOG_DEBUG << "Failed to set elapsed time. "
1646af5d6058SSantosh Puranik                                             "DBUS response error "
1647af5d6058SSantosh Puranik                                          << ec;
1648af5d6058SSantosh Puranik                         messages::internalError(aResp->res);
1649af5d6058SSantosh Puranik                         return;
1650af5d6058SSantosh Puranik                     }
1651af5d6058SSantosh Puranik                     aResp->res.jsonValue["DateTime"] = datetime;
1652af5d6058SSantosh Puranik                 },
1653af5d6058SSantosh Puranik                 "xyz.openbmc_project.Time.Manager",
1654af5d6058SSantosh Puranik                 "/xyz/openbmc_project/time/bmc",
1655af5d6058SSantosh Puranik                 "org.freedesktop.DBus.Properties", "Set",
1656af5d6058SSantosh Puranik                 "xyz.openbmc_project.Time.EpochTime", "Elapsed",
1657af5d6058SSantosh Puranik                 std::variant<uint64_t>(durMicroSecs));
1658af5d6058SSantosh Puranik         }
1659af5d6058SSantosh Puranik         else
1660af5d6058SSantosh Puranik         {
1661af5d6058SSantosh Puranik             messages::propertyValueFormatError(aResp->res, datetime,
1662af5d6058SSantosh Puranik                                                "DateTime");
1663af5d6058SSantosh Puranik             return;
1664af5d6058SSantosh Puranik         }
166583ff9ab6SJames Feist     }
16669c310685SBorawski.Lukasz 
16670f74e643SEd Tanous     std::string uuid;
16689c310685SBorawski.Lukasz };
16699c310685SBorawski.Lukasz 
16701abe55efSEd Tanous class ManagerCollection : public Node
16711abe55efSEd Tanous {
16729c310685SBorawski.Lukasz   public:
16731abe55efSEd Tanous     ManagerCollection(CrowApp& app) : Node(app, "/redfish/v1/Managers/")
16741abe55efSEd Tanous     {
1675a434f2bdSEd Tanous         entityPrivileges = {
1676a434f2bdSEd Tanous             {boost::beast::http::verb::get, {{"Login"}}},
1677e0d918bcSEd Tanous             {boost::beast::http::verb::head, {{"Login"}}},
1678e0d918bcSEd Tanous             {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
1679e0d918bcSEd Tanous             {boost::beast::http::verb::put, {{"ConfigureManager"}}},
1680e0d918bcSEd Tanous             {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
1681e0d918bcSEd Tanous             {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
16829c310685SBorawski.Lukasz     }
16839c310685SBorawski.Lukasz 
16849c310685SBorawski.Lukasz   private:
168555c7b7a2SEd Tanous     void doGet(crow::Response& res, const crow::Request& req,
16861abe55efSEd Tanous                const std::vector<std::string>& params) override
16871abe55efSEd Tanous     {
168883ff9ab6SJames Feist         // Collections don't include the static data added by SubRoute
168983ff9ab6SJames Feist         // because it has a duplicate entry for members
169055c7b7a2SEd Tanous         res.jsonValue["@odata.id"] = "/redfish/v1/Managers";
169155c7b7a2SEd Tanous         res.jsonValue["@odata.type"] = "#ManagerCollection.ManagerCollection";
169255c7b7a2SEd Tanous         res.jsonValue["@odata.context"] =
169355c7b7a2SEd Tanous             "/redfish/v1/$metadata#ManagerCollection.ManagerCollection";
169455c7b7a2SEd Tanous         res.jsonValue["Name"] = "Manager Collection";
169555c7b7a2SEd Tanous         res.jsonValue["Members@odata.count"] = 1;
169655c7b7a2SEd Tanous         res.jsonValue["Members"] = {
16975b4aa86bSJames Feist             {{"@odata.id", "/redfish/v1/Managers/bmc"}}};
16989c310685SBorawski.Lukasz         res.end();
16999c310685SBorawski.Lukasz     }
17009c310685SBorawski.Lukasz };
17019c310685SBorawski.Lukasz } // namespace redfish
1702