xref: /openbmc/bmcweb/features/redfish/lib/managers.hpp (revision 0f74e643ec246c333ef4724af1ecd5adeb1b6658)
19c310685SBorawski.Lukasz /*
29c310685SBorawski.Lukasz // Copyright (c) 2018 Intel Corporation
39c310685SBorawski.Lukasz //
49c310685SBorawski.Lukasz // Licensed under the Apache License, Version 2.0 (the "License");
59c310685SBorawski.Lukasz // you may not use this file except in compliance with the License.
69c310685SBorawski.Lukasz // You may obtain a copy of the License at
79c310685SBorawski.Lukasz //
89c310685SBorawski.Lukasz //      http://www.apache.org/licenses/LICENSE-2.0
99c310685SBorawski.Lukasz //
109c310685SBorawski.Lukasz // Unless required by applicable law or agreed to in writing, software
119c310685SBorawski.Lukasz // distributed under the License is distributed on an "AS IS" BASIS,
129c310685SBorawski.Lukasz // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139c310685SBorawski.Lukasz // See the License for the specific language governing permissions and
149c310685SBorawski.Lukasz // limitations under the License.
159c310685SBorawski.Lukasz */
169c310685SBorawski.Lukasz #pragma once
179c310685SBorawski.Lukasz 
189c310685SBorawski.Lukasz #include "node.hpp"
199c310685SBorawski.Lukasz 
205b4aa86bSJames Feist #include <boost/algorithm/string/replace.hpp>
215b4aa86bSJames Feist #include <dbus_utility.hpp>
225b4aa86bSJames Feist 
231abe55efSEd Tanous namespace redfish
241abe55efSEd Tanous {
25ed5befbdSJennifer Lee 
26ed5befbdSJennifer Lee /**
27ed5befbdSJennifer Lee  * ManagerActionsReset class supports handle POST method for Reset action.
28ed5befbdSJennifer Lee  * The class retrieves and sends data directly to dbus.
29ed5befbdSJennifer Lee  */
30ed5befbdSJennifer Lee class ManagerActionsReset : public Node
31ed5befbdSJennifer Lee {
32ed5befbdSJennifer Lee   public:
33ed5befbdSJennifer Lee     ManagerActionsReset(CrowApp& app) :
34ed5befbdSJennifer Lee         Node(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/")
35ed5befbdSJennifer Lee     {
36ed5befbdSJennifer Lee         entityPrivileges = {
37ed5befbdSJennifer Lee             {boost::beast::http::verb::get, {{"Login"}}},
38ed5befbdSJennifer Lee             {boost::beast::http::verb::head, {{"Login"}}},
39ed5befbdSJennifer Lee             {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
40ed5befbdSJennifer Lee             {boost::beast::http::verb::put, {{"ConfigureManager"}}},
41ed5befbdSJennifer Lee             {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
42ed5befbdSJennifer Lee             {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
43ed5befbdSJennifer Lee     }
44ed5befbdSJennifer Lee 
45ed5befbdSJennifer Lee   private:
46ed5befbdSJennifer Lee     /**
47ed5befbdSJennifer Lee      * Function handles POST method request.
48ed5befbdSJennifer Lee      * Analyzes POST body message before sends Reset request data to dbus.
49ed5befbdSJennifer Lee      * OpenBMC allows for ResetType is GracefulRestart only.
50ed5befbdSJennifer Lee      */
51ed5befbdSJennifer Lee     void doPost(crow::Response& res, const crow::Request& req,
52ed5befbdSJennifer Lee                 const std::vector<std::string>& params) override
53ed5befbdSJennifer Lee     {
54ed5befbdSJennifer Lee         std::string resetType;
55ed5befbdSJennifer Lee 
56ed5befbdSJennifer Lee         if (!json_util::readJson(req, res, "ResetType", resetType))
57ed5befbdSJennifer Lee         {
58ed5befbdSJennifer Lee             return;
59ed5befbdSJennifer Lee         }
60ed5befbdSJennifer Lee 
61ed5befbdSJennifer Lee         if (resetType != "GracefulRestart")
62ed5befbdSJennifer Lee         {
63ed5befbdSJennifer Lee             res.result(boost::beast::http::status::bad_request);
64ed5befbdSJennifer Lee             messages::actionParameterNotSupported(res, resetType, "ResetType");
65ed5befbdSJennifer Lee             BMCWEB_LOG_ERROR << "Request incorrect action parameter: "
66ed5befbdSJennifer Lee                              << resetType;
67ed5befbdSJennifer Lee             res.end();
68ed5befbdSJennifer Lee             return;
69ed5befbdSJennifer Lee         }
70ed5befbdSJennifer Lee         doBMCGracefulRestart(res, req, params);
71ed5befbdSJennifer Lee     }
72ed5befbdSJennifer Lee 
73ed5befbdSJennifer Lee     /**
74ed5befbdSJennifer Lee      * Function transceives data with dbus directly.
75ed5befbdSJennifer Lee      * All BMC state properties will be retrieved before sending reset request.
76ed5befbdSJennifer Lee      */
77ed5befbdSJennifer Lee     void doBMCGracefulRestart(crow::Response& res, const crow::Request& req,
78ed5befbdSJennifer Lee                               const std::vector<std::string>& params)
79ed5befbdSJennifer Lee     {
80ed5befbdSJennifer Lee         const char* processName = "xyz.openbmc_project.State.BMC";
81ed5befbdSJennifer Lee         const char* objectPath = "/xyz/openbmc_project/state/bmc0";
82ed5befbdSJennifer Lee         const char* interfaceName = "xyz.openbmc_project.State.BMC";
83ed5befbdSJennifer Lee         const std::string& propertyValue =
84ed5befbdSJennifer Lee             "xyz.openbmc_project.State.BMC.Transition.Reboot";
85ed5befbdSJennifer Lee         const char* destProperty = "RequestedBMCTransition";
86ed5befbdSJennifer Lee 
87ed5befbdSJennifer Lee         // Create the D-Bus variant for D-Bus call.
88ed5befbdSJennifer Lee         VariantType dbusPropertyValue(propertyValue);
89ed5befbdSJennifer Lee 
90ed5befbdSJennifer Lee         std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res);
91ed5befbdSJennifer Lee 
92ed5befbdSJennifer Lee         crow::connections::systemBus->async_method_call(
93ed5befbdSJennifer Lee             [asyncResp](const boost::system::error_code ec) {
94ed5befbdSJennifer Lee                 // Use "Set" method to set the property value.
95ed5befbdSJennifer Lee                 if (ec)
96ed5befbdSJennifer Lee                 {
97ed5befbdSJennifer Lee                     BMCWEB_LOG_ERROR << "[Set] Bad D-Bus request error: " << ec;
98ed5befbdSJennifer Lee                     messages::internalError(asyncResp->res);
99ed5befbdSJennifer Lee                     return;
100ed5befbdSJennifer Lee                 }
101ed5befbdSJennifer Lee 
102ed5befbdSJennifer Lee                 messages::success(asyncResp->res);
103ed5befbdSJennifer Lee             },
104ed5befbdSJennifer Lee             processName, objectPath, "org.freedesktop.DBus.Properties", "Set",
105ed5befbdSJennifer Lee             interfaceName, destProperty, dbusPropertyValue);
106ed5befbdSJennifer Lee     }
107ed5befbdSJennifer Lee };
108ed5befbdSJennifer Lee 
1095b4aa86bSJames Feist static constexpr const char* objectManagerIface =
1105b4aa86bSJames Feist     "org.freedesktop.DBus.ObjectManager";
1115b4aa86bSJames Feist static constexpr const char* pidConfigurationIface =
1125b4aa86bSJames Feist     "xyz.openbmc_project.Configuration.Pid";
1135b4aa86bSJames Feist static constexpr const char* pidZoneConfigurationIface =
1145b4aa86bSJames Feist     "xyz.openbmc_project.Configuration.Pid.Zone";
1159c310685SBorawski.Lukasz 
1165b4aa86bSJames Feist static void asyncPopulatePid(const std::string& connection,
1175b4aa86bSJames Feist                              const std::string& path,
1185b4aa86bSJames Feist                              std::shared_ptr<AsyncResp> asyncResp)
1195b4aa86bSJames Feist {
1205b4aa86bSJames Feist 
1215b4aa86bSJames Feist     crow::connections::systemBus->async_method_call(
1225b4aa86bSJames Feist         [asyncResp](const boost::system::error_code ec,
1235b4aa86bSJames Feist                     const dbus::utility::ManagedObjectType& managedObj) {
1245b4aa86bSJames Feist             if (ec)
1255b4aa86bSJames Feist             {
1265b4aa86bSJames Feist                 BMCWEB_LOG_ERROR << ec;
1275b4aa86bSJames Feist                 asyncResp->res.jsonValue.clear();
128f12894f8SJason M. Bills                 messages::internalError(asyncResp->res);
1295b4aa86bSJames Feist                 return;
1305b4aa86bSJames Feist             }
1315b4aa86bSJames Feist             nlohmann::json& configRoot =
1325b4aa86bSJames Feist                 asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"];
1335b4aa86bSJames Feist             nlohmann::json& fans = configRoot["FanControllers"];
1345b4aa86bSJames Feist             fans["@odata.type"] = "#OemManager.FanControllers";
1355b4aa86bSJames Feist             fans["@odata.context"] =
1365b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.FanControllers";
1375b4aa86bSJames Feist             fans["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/"
1385b4aa86bSJames Feist                                 "Fan/FanControllers";
1395b4aa86bSJames Feist 
1405b4aa86bSJames Feist             nlohmann::json& pids = configRoot["PidControllers"];
1415b4aa86bSJames Feist             pids["@odata.type"] = "#OemManager.PidControllers";
1425b4aa86bSJames Feist             pids["@odata.context"] =
1435b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.PidControllers";
1445b4aa86bSJames Feist             pids["@odata.id"] =
1455b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers";
1465b4aa86bSJames Feist 
1475b4aa86bSJames Feist             nlohmann::json& zones = configRoot["FanZones"];
1485b4aa86bSJames Feist             zones["@odata.id"] =
1495b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones";
1505b4aa86bSJames Feist             zones["@odata.type"] = "#OemManager.FanZones";
1515b4aa86bSJames Feist             zones["@odata.context"] =
1525b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.FanZones";
1535b4aa86bSJames Feist             configRoot["@odata.id"] =
1545b4aa86bSJames Feist                 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan";
1555b4aa86bSJames Feist             configRoot["@odata.type"] = "#OemManager.Fan";
1565b4aa86bSJames Feist             configRoot["@odata.context"] =
1575b4aa86bSJames Feist                 "/redfish/v1/$metadata#OemManager.Fan";
1585b4aa86bSJames Feist 
1595b4aa86bSJames Feist             bool propertyError = false;
1605b4aa86bSJames Feist             for (const auto& pathPair : managedObj)
1615b4aa86bSJames Feist             {
1625b4aa86bSJames Feist                 for (const auto& intfPair : pathPair.second)
1635b4aa86bSJames Feist                 {
1645b4aa86bSJames Feist                     if (intfPair.first != pidConfigurationIface &&
1655b4aa86bSJames Feist                         intfPair.first != pidZoneConfigurationIface)
1665b4aa86bSJames Feist                     {
1675b4aa86bSJames Feist                         continue;
1685b4aa86bSJames Feist                     }
1695b4aa86bSJames Feist                     auto findName = intfPair.second.find("Name");
1705b4aa86bSJames Feist                     if (findName == intfPair.second.end())
1715b4aa86bSJames Feist                     {
1725b4aa86bSJames Feist                         BMCWEB_LOG_ERROR << "Pid Field missing Name";
173a08b46ccSJason M. Bills                         messages::internalError(asyncResp->res);
1745b4aa86bSJames Feist                         return;
1755b4aa86bSJames Feist                     }
1765b4aa86bSJames Feist                     const std::string* namePtr =
1775b4aa86bSJames Feist                         mapbox::getPtr<const std::string>(findName->second);
1785b4aa86bSJames Feist                     if (namePtr == nullptr)
1795b4aa86bSJames Feist                     {
1805b4aa86bSJames Feist                         BMCWEB_LOG_ERROR << "Pid Name Field illegal";
1815b4aa86bSJames Feist                         return;
1825b4aa86bSJames Feist                     }
1835b4aa86bSJames Feist 
1845b4aa86bSJames Feist                     std::string name = *namePtr;
1855b4aa86bSJames Feist                     dbus::utility::escapePathForDbus(name);
1865b4aa86bSJames Feist                     if (intfPair.first == pidZoneConfigurationIface)
1875b4aa86bSJames Feist                     {
1885b4aa86bSJames Feist                         std::string chassis;
1895b4aa86bSJames Feist                         if (!dbus::utility::getNthStringFromPath(
1905b4aa86bSJames Feist                                 pathPair.first.str, 5, chassis))
1915b4aa86bSJames Feist                         {
1925b4aa86bSJames Feist                             chassis = "#IllegalValue";
1935b4aa86bSJames Feist                         }
1945b4aa86bSJames Feist                         nlohmann::json& zone = zones[name];
1955b4aa86bSJames Feist                         zone["Chassis"] = {
1965b4aa86bSJames Feist                             {"@odata.id", "/redfish/v1/Chassis/" + chassis}};
1975b4aa86bSJames Feist                         zone["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/"
1985b4aa86bSJames Feist                                             "OpenBmc/Fan/FanZones/" +
1995b4aa86bSJames Feist                                             name;
2005b4aa86bSJames Feist                         zone["@odata.type"] = "#OemManager.FanZone";
2015b4aa86bSJames Feist                         zone["@odata.context"] =
2025b4aa86bSJames Feist                             "/redfish/v1/$metadata#OemManager.FanZone";
2035b4aa86bSJames Feist                     }
2045b4aa86bSJames Feist 
2055b4aa86bSJames Feist                     for (const auto& propertyPair : intfPair.second)
2065b4aa86bSJames Feist                     {
2075b4aa86bSJames Feist                         if (propertyPair.first == "Type" ||
2085b4aa86bSJames Feist                             propertyPair.first == "Class" ||
2095b4aa86bSJames Feist                             propertyPair.first == "Name")
2105b4aa86bSJames Feist                         {
2115b4aa86bSJames Feist                             continue;
2125b4aa86bSJames Feist                         }
2135b4aa86bSJames Feist 
2145b4aa86bSJames Feist                         // zones
2155b4aa86bSJames Feist                         if (intfPair.first == pidZoneConfigurationIface)
2165b4aa86bSJames Feist                         {
2175b4aa86bSJames Feist                             const double* ptr = mapbox::getPtr<const double>(
2185b4aa86bSJames Feist                                 propertyPair.second);
2195b4aa86bSJames Feist                             if (ptr == nullptr)
2205b4aa86bSJames Feist                             {
2215b4aa86bSJames Feist                                 BMCWEB_LOG_ERROR << "Field Illegal "
2225b4aa86bSJames Feist                                                  << propertyPair.first;
223f12894f8SJason M. Bills                                 messages::internalError(asyncResp->res);
2245b4aa86bSJames Feist                                 return;
2255b4aa86bSJames Feist                             }
2265b4aa86bSJames Feist                             zones[name][propertyPair.first] = *ptr;
2275b4aa86bSJames Feist                         }
2285b4aa86bSJames Feist 
2295b4aa86bSJames Feist                         // pid and fans are off the same configuration
2305b4aa86bSJames Feist                         if (intfPair.first == pidConfigurationIface)
2315b4aa86bSJames Feist                         {
2325b4aa86bSJames Feist                             const std::string* classPtr = nullptr;
2335b4aa86bSJames Feist                             auto findClass = intfPair.second.find("Class");
2345b4aa86bSJames Feist                             if (findClass != intfPair.second.end())
2355b4aa86bSJames Feist                             {
2365b4aa86bSJames Feist                                 classPtr = mapbox::getPtr<const std::string>(
2375b4aa86bSJames Feist                                     findClass->second);
2385b4aa86bSJames Feist                             }
2395b4aa86bSJames Feist                             if (classPtr == nullptr)
2405b4aa86bSJames Feist                             {
2415b4aa86bSJames Feist                                 BMCWEB_LOG_ERROR << "Pid Class Field illegal";
242a08b46ccSJason M. Bills                                 messages::internalError(asyncResp->res);
2435b4aa86bSJames Feist                                 return;
2445b4aa86bSJames Feist                             }
2455b4aa86bSJames Feist                             bool isFan = *classPtr == "fan";
2465b4aa86bSJames Feist                             nlohmann::json& element =
2475b4aa86bSJames Feist                                 isFan ? fans[name] : pids[name];
2485b4aa86bSJames Feist                             if (isFan)
2495b4aa86bSJames Feist                             {
2505b4aa86bSJames Feist                                 element["@odata.id"] =
2515b4aa86bSJames Feist                                     "/redfish/v1/Managers/bmc#/Oem/"
2525b4aa86bSJames Feist                                     "OpenBmc/Fan/FanControllers/" +
2535b4aa86bSJames Feist                                     std::string(name);
2545b4aa86bSJames Feist                                 element["@odata.type"] =
2555b4aa86bSJames Feist                                     "#OemManager.FanController";
2565b4aa86bSJames Feist 
2575b4aa86bSJames Feist                                 element["@odata.context"] =
2585b4aa86bSJames Feist                                     "/redfish/v1/"
2595b4aa86bSJames Feist                                     "$metadata#OemManager.FanController";
2605b4aa86bSJames Feist                             }
2615b4aa86bSJames Feist                             else
2625b4aa86bSJames Feist                             {
2635b4aa86bSJames Feist                                 element["@odata.id"] =
2645b4aa86bSJames Feist                                     "/redfish/v1/Managers/bmc#/Oem/"
2655b4aa86bSJames Feist                                     "OpenBmc/Fan/PidControllers/" +
2665b4aa86bSJames Feist                                     std::string(name);
2675b4aa86bSJames Feist                                 element["@odata.type"] =
2685b4aa86bSJames Feist                                     "#OemManager.PidController";
2695b4aa86bSJames Feist                                 element["@odata.context"] =
2705b4aa86bSJames Feist                                     "/redfish/v1/$metadata"
2715b4aa86bSJames Feist                                     "#OemManager.PidController";
2725b4aa86bSJames Feist                             }
2735b4aa86bSJames Feist 
2745b4aa86bSJames Feist                             if (propertyPair.first == "Zones")
2755b4aa86bSJames Feist                             {
2765b4aa86bSJames Feist                                 const std::vector<std::string>* inputs =
2775b4aa86bSJames Feist                                     mapbox::getPtr<
2785b4aa86bSJames Feist                                         const std::vector<std::string>>(
2795b4aa86bSJames Feist                                         propertyPair.second);
2805b4aa86bSJames Feist 
2815b4aa86bSJames Feist                                 if (inputs == nullptr)
2825b4aa86bSJames Feist                                 {
2835b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR
2845b4aa86bSJames Feist                                         << "Zones Pid Field Illegal";
285a08b46ccSJason M. Bills                                     messages::internalError(asyncResp->res);
2865b4aa86bSJames Feist                                     return;
2875b4aa86bSJames Feist                                 }
2885b4aa86bSJames Feist                                 auto& data = element[propertyPair.first];
2895b4aa86bSJames Feist                                 data = nlohmann::json::array();
2905b4aa86bSJames Feist                                 for (std::string itemCopy : *inputs)
2915b4aa86bSJames Feist                                 {
2925b4aa86bSJames Feist                                     dbus::utility::escapePathForDbus(itemCopy);
2935b4aa86bSJames Feist                                     data.push_back(
2945b4aa86bSJames Feist                                         {{"@odata.id",
2955b4aa86bSJames Feist                                           "/redfish/v1/Managers/bmc#/Oem/"
2965b4aa86bSJames Feist                                           "OpenBmc/Fan/FanZones/" +
2975b4aa86bSJames Feist                                               itemCopy}});
2985b4aa86bSJames Feist                                 }
2995b4aa86bSJames Feist                             }
3005b4aa86bSJames Feist                             // todo(james): may never happen, but this
3015b4aa86bSJames Feist                             // assumes configuration data referenced in the
3025b4aa86bSJames Feist                             // PID config is provided by the same daemon, we
3035b4aa86bSJames Feist                             // could add another loop to cover all cases,
3045b4aa86bSJames Feist                             // but I'm okay kicking this can down the road a
3055b4aa86bSJames Feist                             // bit
3065b4aa86bSJames Feist 
3075b4aa86bSJames Feist                             else if (propertyPair.first == "Inputs" ||
3085b4aa86bSJames Feist                                      propertyPair.first == "Outputs")
3095b4aa86bSJames Feist                             {
3105b4aa86bSJames Feist                                 auto& data = element[propertyPair.first];
3115b4aa86bSJames Feist                                 const std::vector<std::string>* inputs =
3125b4aa86bSJames Feist                                     mapbox::getPtr<
3135b4aa86bSJames Feist                                         const std::vector<std::string>>(
3145b4aa86bSJames Feist                                         propertyPair.second);
3155b4aa86bSJames Feist 
3165b4aa86bSJames Feist                                 if (inputs == nullptr)
3175b4aa86bSJames Feist                                 {
3185b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
3195b4aa86bSJames Feist                                                      << propertyPair.first;
320f12894f8SJason M. Bills                                     messages::internalError(asyncResp->res);
3215b4aa86bSJames Feist                                     return;
3225b4aa86bSJames Feist                                 }
3235b4aa86bSJames Feist                                 data = *inputs;
3245b4aa86bSJames Feist                             } // doubles
3255b4aa86bSJames Feist                             else if (propertyPair.first ==
3265b4aa86bSJames Feist                                          "FFGainCoefficient" ||
3275b4aa86bSJames Feist                                      propertyPair.first == "FFOffCoefficient" ||
3285b4aa86bSJames Feist                                      propertyPair.first == "ICoefficient" ||
3295b4aa86bSJames Feist                                      propertyPair.first == "ILimitMax" ||
3305b4aa86bSJames Feist                                      propertyPair.first == "ILimitMin" ||
3315b4aa86bSJames Feist                                      propertyPair.first == "OutLimitMax" ||
3325b4aa86bSJames Feist                                      propertyPair.first == "OutLimitMin" ||
3335b4aa86bSJames Feist                                      propertyPair.first == "PCoefficient" ||
3345b4aa86bSJames Feist                                      propertyPair.first == "SlewNeg" ||
3355b4aa86bSJames Feist                                      propertyPair.first == "SlewPos")
3365b4aa86bSJames Feist                             {
3375b4aa86bSJames Feist                                 const double* ptr =
3385b4aa86bSJames Feist                                     mapbox::getPtr<const double>(
3395b4aa86bSJames Feist                                         propertyPair.second);
3405b4aa86bSJames Feist                                 if (ptr == nullptr)
3415b4aa86bSJames Feist                                 {
3425b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
3435b4aa86bSJames Feist                                                      << propertyPair.first;
344f12894f8SJason M. Bills                                     messages::internalError(asyncResp->res);
3455b4aa86bSJames Feist                                     return;
3465b4aa86bSJames Feist                                 }
3475b4aa86bSJames Feist                                 element[propertyPair.first] = *ptr;
3485b4aa86bSJames Feist                             }
3495b4aa86bSJames Feist                         }
3505b4aa86bSJames Feist                     }
3515b4aa86bSJames Feist                 }
3525b4aa86bSJames Feist             }
3535b4aa86bSJames Feist         },
3545b4aa86bSJames Feist         connection, path, objectManagerIface, "GetManagedObjects");
3555b4aa86bSJames Feist }
356ca537928SJennifer Lee 
35783ff9ab6SJames Feist enum class CreatePIDRet
35883ff9ab6SJames Feist {
35983ff9ab6SJames Feist     fail,
36083ff9ab6SJames Feist     del,
36183ff9ab6SJames Feist     patch
36283ff9ab6SJames Feist };
36383ff9ab6SJames Feist 
36483ff9ab6SJames Feist static CreatePIDRet createPidInterface(
36583ff9ab6SJames Feist     const std::shared_ptr<AsyncResp>& response, const std::string& type,
36683ff9ab6SJames Feist     const nlohmann::json& record, const std::string& path,
36783ff9ab6SJames Feist     const dbus::utility::ManagedObjectType& managedObj, bool createNewObject,
36883ff9ab6SJames Feist     boost::container::flat_map<std::string, dbus::utility::DbusVariantType>&
36983ff9ab6SJames Feist         output,
37083ff9ab6SJames Feist     std::string& chassis)
37183ff9ab6SJames Feist {
37283ff9ab6SJames Feist 
37383ff9ab6SJames Feist     if (type == "PidControllers" || type == "FanControllers")
37483ff9ab6SJames Feist     {
37583ff9ab6SJames Feist         if (createNewObject)
37683ff9ab6SJames Feist         {
37783ff9ab6SJames Feist             output["Class"] = type == "PidControllers" ? std::string("temp")
37883ff9ab6SJames Feist                                                        : std::string("fan");
37983ff9ab6SJames Feist             output["Type"] = std::string("Pid");
38083ff9ab6SJames Feist         }
38183ff9ab6SJames Feist         else if (record == nullptr)
38283ff9ab6SJames Feist         {
38383ff9ab6SJames Feist             // delete interface
38483ff9ab6SJames Feist             crow::connections::systemBus->async_method_call(
38583ff9ab6SJames Feist                 [response,
38683ff9ab6SJames Feist                  path{std::string(path)}](const boost::system::error_code ec) {
38783ff9ab6SJames Feist                     if (ec)
38883ff9ab6SJames Feist                     {
38983ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Error patching " << path << ": "
39083ff9ab6SJames Feist                                          << ec;
39135a62c7cSJason M. Bills                         messages::internalError(response->res);
39283ff9ab6SJames Feist                     }
39383ff9ab6SJames Feist                 },
39483ff9ab6SJames Feist                 "xyz.openbmc_project.EntityManager", path,
39583ff9ab6SJames Feist                 pidConfigurationIface, "Delete");
39683ff9ab6SJames Feist             return CreatePIDRet::del;
39783ff9ab6SJames Feist         }
39883ff9ab6SJames Feist 
39983ff9ab6SJames Feist         for (auto& field : record.items())
40083ff9ab6SJames Feist         {
40183ff9ab6SJames Feist             if (field.key() == "Zones")
40283ff9ab6SJames Feist             {
40383ff9ab6SJames Feist                 if (!field.value().is_array())
40483ff9ab6SJames Feist                 {
40583ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
40635a62c7cSJason M. Bills                     messages::propertyValueFormatError(
40735a62c7cSJason M. Bills                         response->res, field.value(), field.key());
40883ff9ab6SJames Feist                     return CreatePIDRet::fail;
40983ff9ab6SJames Feist                 }
41083ff9ab6SJames Feist                 std::vector<std::string> inputs;
41183ff9ab6SJames Feist                 for (const auto& odata : field.value().items())
41283ff9ab6SJames Feist                 {
41383ff9ab6SJames Feist                     for (const auto& value : odata.value().items())
41483ff9ab6SJames Feist                     {
41583ff9ab6SJames Feist                         const std::string* path =
41683ff9ab6SJames Feist                             value.value().get_ptr<const std::string*>();
41783ff9ab6SJames Feist                         if (path == nullptr)
41883ff9ab6SJames Feist                         {
41983ff9ab6SJames Feist                             BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
42083ff9ab6SJames Feist                             messages::propertyValueFormatError(
42135a62c7cSJason M. Bills                                 response->res, field.value().dump(),
42235a62c7cSJason M. Bills                                 field.key());
42383ff9ab6SJames Feist                             return CreatePIDRet::fail;
42483ff9ab6SJames Feist                         }
42583ff9ab6SJames Feist                         std::string input;
42683ff9ab6SJames Feist                         if (!dbus::utility::getNthStringFromPath(*path, 4,
42783ff9ab6SJames Feist                                                                  input))
42883ff9ab6SJames Feist                         {
42983ff9ab6SJames Feist                             BMCWEB_LOG_ERROR << "Got invalid path " << *path;
43083ff9ab6SJames Feist                             messages::propertyValueFormatError(
43135a62c7cSJason M. Bills                                 response->res, field.value().dump(),
43235a62c7cSJason M. Bills                                 field.key());
43383ff9ab6SJames Feist                             return CreatePIDRet::fail;
43483ff9ab6SJames Feist                         }
43583ff9ab6SJames Feist                         boost::replace_all(input, "_", " ");
43683ff9ab6SJames Feist                         inputs.emplace_back(std::move(input));
43783ff9ab6SJames Feist                     }
43883ff9ab6SJames Feist                 }
43983ff9ab6SJames Feist                 output["Zones"] = std::move(inputs);
44083ff9ab6SJames Feist             }
44183ff9ab6SJames Feist             else if (field.key() == "Inputs" || field.key() == "Outputs")
44283ff9ab6SJames Feist             {
44383ff9ab6SJames Feist                 if (!field.value().is_array())
44483ff9ab6SJames Feist                 {
44583ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
44635a62c7cSJason M. Bills                     messages::propertyValueFormatError(
44735a62c7cSJason M. Bills                         response->res, field.value().dump(), field.key());
44883ff9ab6SJames Feist                     return CreatePIDRet::fail;
44983ff9ab6SJames Feist                 }
45083ff9ab6SJames Feist                 std::vector<std::string> inputs;
45183ff9ab6SJames Feist                 for (const auto& value : field.value().items())
45283ff9ab6SJames Feist                 {
45383ff9ab6SJames Feist                     const std::string* sensor =
45483ff9ab6SJames Feist                         value.value().get_ptr<const std::string*>();
45583ff9ab6SJames Feist 
45683ff9ab6SJames Feist                     if (sensor == nullptr)
45783ff9ab6SJames Feist                     {
45883ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Illegal Type "
45983ff9ab6SJames Feist                                          << field.value().dump();
46083ff9ab6SJames Feist                         messages::propertyValueFormatError(
46135a62c7cSJason M. Bills                             response->res, field.value().dump(), field.key());
46283ff9ab6SJames Feist                         return CreatePIDRet::fail;
46383ff9ab6SJames Feist                     }
46483ff9ab6SJames Feist 
46583ff9ab6SJames Feist                     std::string input =
46683ff9ab6SJames Feist                         boost::replace_all_copy(*sensor, "_", " ");
46783ff9ab6SJames Feist                     inputs.push_back(std::move(input));
46883ff9ab6SJames Feist                     // try to find the sensor in the
46983ff9ab6SJames Feist                     // configuration
47083ff9ab6SJames Feist                     if (chassis.empty())
47183ff9ab6SJames Feist                     {
47283ff9ab6SJames Feist                         std::find_if(
47383ff9ab6SJames Feist                             managedObj.begin(), managedObj.end(),
47483ff9ab6SJames Feist                             [&chassis, sensor](const auto& obj) {
47583ff9ab6SJames Feist                                 if (boost::algorithm::ends_with(obj.first.str,
47683ff9ab6SJames Feist                                                                 *sensor))
47783ff9ab6SJames Feist                                 {
47883ff9ab6SJames Feist                                     return dbus::utility::getNthStringFromPath(
47983ff9ab6SJames Feist                                         obj.first.str, 5, chassis);
48083ff9ab6SJames Feist                                 }
48183ff9ab6SJames Feist                                 return false;
48283ff9ab6SJames Feist                             });
48383ff9ab6SJames Feist                     }
48483ff9ab6SJames Feist                 }
48583ff9ab6SJames Feist                 output[field.key()] = inputs;
48683ff9ab6SJames Feist             }
48783ff9ab6SJames Feist 
48883ff9ab6SJames Feist             // doubles
48983ff9ab6SJames Feist             else if (field.key() == "FFGainCoefficient" ||
49083ff9ab6SJames Feist                      field.key() == "FFOffCoefficient" ||
49183ff9ab6SJames Feist                      field.key() == "ICoefficient" ||
49283ff9ab6SJames Feist                      field.key() == "ILimitMax" || field.key() == "ILimitMin" ||
49383ff9ab6SJames Feist                      field.key() == "OutLimitMax" ||
49483ff9ab6SJames Feist                      field.key() == "OutLimitMin" ||
49583ff9ab6SJames Feist                      field.key() == "PCoefficient" ||
49683ff9ab6SJames Feist                      field.key() == "SetPoint" || field.key() == "SlewNeg" ||
49783ff9ab6SJames Feist                      field.key() == "SlewPos")
49883ff9ab6SJames Feist             {
49983ff9ab6SJames Feist                 const double* ptr = field.value().get_ptr<const double*>();
50083ff9ab6SJames Feist                 if (ptr == nullptr)
50183ff9ab6SJames Feist                 {
50283ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
50335a62c7cSJason M. Bills                     messages::propertyValueFormatError(
50435a62c7cSJason M. Bills                         response->res, field.value().dump(), field.key());
50583ff9ab6SJames Feist                     return CreatePIDRet::fail;
50683ff9ab6SJames Feist                 }
50783ff9ab6SJames Feist                 output[field.key()] = *ptr;
50883ff9ab6SJames Feist             }
50983ff9ab6SJames Feist 
51083ff9ab6SJames Feist             else
51183ff9ab6SJames Feist             {
51283ff9ab6SJames Feist                 BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
51335a62c7cSJason M. Bills                 messages::propertyUnknown(response->res, field.key());
51483ff9ab6SJames Feist                 return CreatePIDRet::fail;
51583ff9ab6SJames Feist             }
51683ff9ab6SJames Feist         }
51783ff9ab6SJames Feist     }
51883ff9ab6SJames Feist     else if (type == "FanZones")
51983ff9ab6SJames Feist     {
52083ff9ab6SJames Feist         if (!createNewObject && record == nullptr)
52183ff9ab6SJames Feist         {
52283ff9ab6SJames Feist             // delete interface
52383ff9ab6SJames Feist             crow::connections::systemBus->async_method_call(
52483ff9ab6SJames Feist                 [response,
52583ff9ab6SJames Feist                  path{std::string(path)}](const boost::system::error_code ec) {
52683ff9ab6SJames Feist                     if (ec)
52783ff9ab6SJames Feist                     {
52883ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Error patching " << path << ": "
52983ff9ab6SJames Feist                                          << ec;
53035a62c7cSJason M. Bills                         messages::internalError(response->res);
53183ff9ab6SJames Feist                     }
53283ff9ab6SJames Feist                 },
53383ff9ab6SJames Feist                 "xyz.openbmc_project.EntityManager", path,
53483ff9ab6SJames Feist                 pidZoneConfigurationIface, "Delete");
53583ff9ab6SJames Feist             return CreatePIDRet::del;
53683ff9ab6SJames Feist         }
53783ff9ab6SJames Feist         output["Type"] = std::string("Pid.Zone");
53883ff9ab6SJames Feist 
53983ff9ab6SJames Feist         for (auto& field : record.items())
54083ff9ab6SJames Feist         {
54183ff9ab6SJames Feist             if (field.key() == "Chassis")
54283ff9ab6SJames Feist             {
54383ff9ab6SJames Feist                 const std::string* chassisId = nullptr;
54483ff9ab6SJames Feist                 for (const auto& id : field.value().items())
54583ff9ab6SJames Feist                 {
54683ff9ab6SJames Feist                     if (id.key() != "@odata.id")
54783ff9ab6SJames Feist                     {
54883ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Illegal Type " << id.key();
54935a62c7cSJason M. Bills                         messages::propertyUnknown(response->res, field.key());
55083ff9ab6SJames Feist                         return CreatePIDRet::fail;
55183ff9ab6SJames Feist                     }
55283ff9ab6SJames Feist                     chassisId = id.value().get_ptr<const std::string*>();
55383ff9ab6SJames Feist                     if (chassisId == nullptr)
55483ff9ab6SJames Feist                     {
55583ff9ab6SJames Feist                         messages::createFailedMissingReqProperties(
55635a62c7cSJason M. Bills                             response->res, field.key());
55783ff9ab6SJames Feist                         return CreatePIDRet::fail;
55883ff9ab6SJames Feist                     }
55983ff9ab6SJames Feist                 }
56083ff9ab6SJames Feist 
56183ff9ab6SJames Feist                 // /refish/v1/chassis/chassis_name/
56283ff9ab6SJames Feist                 if (!dbus::utility::getNthStringFromPath(*chassisId, 3,
56383ff9ab6SJames Feist                                                          chassis))
56483ff9ab6SJames Feist                 {
56583ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Got invalid path " << *chassisId;
56635a62c7cSJason M. Bills                     messages::invalidObject(response->res, *chassisId);
56783ff9ab6SJames Feist                     return CreatePIDRet::fail;
56883ff9ab6SJames Feist                 }
56983ff9ab6SJames Feist             }
57083ff9ab6SJames Feist             else if (field.key() == "FailSafePercent" ||
57183ff9ab6SJames Feist                      field.key() == "MinThermalRpm")
57283ff9ab6SJames Feist             {
57383ff9ab6SJames Feist                 const double* ptr = field.value().get_ptr<const double*>();
57483ff9ab6SJames Feist                 if (ptr == nullptr)
57583ff9ab6SJames Feist                 {
57683ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
57735a62c7cSJason M. Bills                     messages::propertyValueFormatError(
57835a62c7cSJason M. Bills                         response->res, field.value().dump(), field.key());
57983ff9ab6SJames Feist                     return CreatePIDRet::fail;
58083ff9ab6SJames Feist                 }
58183ff9ab6SJames Feist                 output[field.key()] = *ptr;
58283ff9ab6SJames Feist             }
58383ff9ab6SJames Feist             else
58483ff9ab6SJames Feist             {
58583ff9ab6SJames Feist                 BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
58635a62c7cSJason M. Bills                 messages::propertyUnknown(response->res, field.key());
58783ff9ab6SJames Feist                 return CreatePIDRet::fail;
58883ff9ab6SJames Feist             }
58983ff9ab6SJames Feist         }
59083ff9ab6SJames Feist     }
59183ff9ab6SJames Feist     else
59283ff9ab6SJames Feist     {
59383ff9ab6SJames Feist         BMCWEB_LOG_ERROR << "Illegal Type " << type;
59435a62c7cSJason M. Bills         messages::propertyUnknown(response->res, type);
59583ff9ab6SJames Feist         return CreatePIDRet::fail;
59683ff9ab6SJames Feist     }
59783ff9ab6SJames Feist     return CreatePIDRet::patch;
59883ff9ab6SJames Feist }
59983ff9ab6SJames Feist 
6001abe55efSEd Tanous class Manager : public Node
6011abe55efSEd Tanous {
6029c310685SBorawski.Lukasz   public:
6035b4aa86bSJames Feist     Manager(CrowApp& app) : Node(app, "/redfish/v1/Managers/bmc/")
6041abe55efSEd Tanous     {
605*0f74e643SEd Tanous         uuid = app.template getMiddleware<crow::persistent_data::Middleware>()
60655c7b7a2SEd Tanous                    .systemUuid;
607a434f2bdSEd Tanous         entityPrivileges = {
608a434f2bdSEd Tanous             {boost::beast::http::verb::get, {{"Login"}}},
609e0d918bcSEd Tanous             {boost::beast::http::verb::head, {{"Login"}}},
610e0d918bcSEd Tanous             {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
611e0d918bcSEd Tanous             {boost::beast::http::verb::put, {{"ConfigureManager"}}},
612e0d918bcSEd Tanous             {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
613e0d918bcSEd Tanous             {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
6149c310685SBorawski.Lukasz     }
6159c310685SBorawski.Lukasz 
6169c310685SBorawski.Lukasz   private:
6175b4aa86bSJames Feist     void getPidValues(std::shared_ptr<AsyncResp> asyncResp)
6185b4aa86bSJames Feist     {
6195b4aa86bSJames Feist         crow::connections::systemBus->async_method_call(
6205b4aa86bSJames Feist             [asyncResp](const boost::system::error_code ec,
6215b4aa86bSJames Feist                         const crow::openbmc_mapper::GetSubTreeType& subtree) {
6225b4aa86bSJames Feist                 if (ec)
6235b4aa86bSJames Feist                 {
6245b4aa86bSJames Feist                     BMCWEB_LOG_ERROR << ec;
625f12894f8SJason M. Bills                     messages::internalError(asyncResp->res);
6265b4aa86bSJames Feist                     return;
6275b4aa86bSJames Feist                 }
6285b4aa86bSJames Feist 
6295b4aa86bSJames Feist                 // create map of <connection, path to objMgr>>
6305b4aa86bSJames Feist                 boost::container::flat_map<std::string, std::string>
6315b4aa86bSJames Feist                     objectMgrPaths;
6326bce33bcSJames Feist                 boost::container::flat_set<std::string> calledConnections;
6335b4aa86bSJames Feist                 for (const auto& pathGroup : subtree)
6345b4aa86bSJames Feist                 {
6355b4aa86bSJames Feist                     for (const auto& connectionGroup : pathGroup.second)
6365b4aa86bSJames Feist                     {
6376bce33bcSJames Feist                         auto findConnection =
6386bce33bcSJames Feist                             calledConnections.find(connectionGroup.first);
6396bce33bcSJames Feist                         if (findConnection != calledConnections.end())
6406bce33bcSJames Feist                         {
6416bce33bcSJames Feist                             break;
6426bce33bcSJames Feist                         }
6435b4aa86bSJames Feist                         for (const std::string& interface :
6445b4aa86bSJames Feist                              connectionGroup.second)
6455b4aa86bSJames Feist                         {
6465b4aa86bSJames Feist                             if (interface == objectManagerIface)
6475b4aa86bSJames Feist                             {
6485b4aa86bSJames Feist                                 objectMgrPaths[connectionGroup.first] =
6495b4aa86bSJames Feist                                     pathGroup.first;
6505b4aa86bSJames Feist                             }
6515b4aa86bSJames Feist                             // this list is alphabetical, so we
6525b4aa86bSJames Feist                             // should have found the objMgr by now
6535b4aa86bSJames Feist                             if (interface == pidConfigurationIface ||
6545b4aa86bSJames Feist                                 interface == pidZoneConfigurationIface)
6555b4aa86bSJames Feist                             {
6565b4aa86bSJames Feist                                 auto findObjMgr =
6575b4aa86bSJames Feist                                     objectMgrPaths.find(connectionGroup.first);
6585b4aa86bSJames Feist                                 if (findObjMgr == objectMgrPaths.end())
6595b4aa86bSJames Feist                                 {
6605b4aa86bSJames Feist                                     BMCWEB_LOG_DEBUG << connectionGroup.first
6615b4aa86bSJames Feist                                                      << "Has no Object Manager";
6625b4aa86bSJames Feist                                     continue;
6635b4aa86bSJames Feist                                 }
6646bce33bcSJames Feist 
6656bce33bcSJames Feist                                 calledConnections.insert(connectionGroup.first);
6666bce33bcSJames Feist 
6675b4aa86bSJames Feist                                 asyncPopulatePid(findObjMgr->first,
6685b4aa86bSJames Feist                                                  findObjMgr->second, asyncResp);
6695b4aa86bSJames Feist                                 break;
6705b4aa86bSJames Feist                             }
6715b4aa86bSJames Feist                         }
6725b4aa86bSJames Feist                     }
6735b4aa86bSJames Feist                 }
6745b4aa86bSJames Feist             },
6755b4aa86bSJames Feist             "xyz.openbmc_project.ObjectMapper",
6765b4aa86bSJames Feist             "/xyz/openbmc_project/object_mapper",
6775b4aa86bSJames Feist             "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0,
6785b4aa86bSJames Feist             std::array<const char*, 3>{pidConfigurationIface,
6795b4aa86bSJames Feist                                        pidZoneConfigurationIface,
6805b4aa86bSJames Feist                                        objectManagerIface});
6815b4aa86bSJames Feist     }
6825b4aa86bSJames Feist 
68355c7b7a2SEd Tanous     void doGet(crow::Response& res, const crow::Request& req,
6841abe55efSEd Tanous                const std::vector<std::string>& params) override
6851abe55efSEd Tanous     {
686*0f74e643SEd Tanous         res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc";
687*0f74e643SEd Tanous         res.jsonValue["@odata.type"] = "#Manager.v1_3_0.Manager";
688*0f74e643SEd Tanous         res.jsonValue["@odata.context"] =
689*0f74e643SEd Tanous             "/redfish/v1/$metadata#Manager.Manager";
690*0f74e643SEd Tanous         res.jsonValue["Id"] = "bmc";
691*0f74e643SEd Tanous         res.jsonValue["Name"] = "OpenBmc Manager";
692*0f74e643SEd Tanous         res.jsonValue["Description"] = "Baseboard Management Controller";
693*0f74e643SEd Tanous         res.jsonValue["PowerState"] = "On";
694*0f74e643SEd Tanous         res.jsonValue["ManagerType"] = "BMC";
695*0f74e643SEd Tanous         res.jsonValue["UUID"] =
696*0f74e643SEd Tanous 
697*0f74e643SEd Tanous             res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model
698*0f74e643SEd Tanous 
699*0f74e643SEd Tanous         res.jsonValue["LogServices"] = {
700*0f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/LogServices"}};
701*0f74e643SEd Tanous 
702*0f74e643SEd Tanous         res.jsonValue["NetworkProtocol"] = {
703*0f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/NetworkProtocol"}};
704*0f74e643SEd Tanous 
705*0f74e643SEd Tanous         res.jsonValue["EthernetInterfaces"] = {
706*0f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/EthernetInterfaces"}};
707*0f74e643SEd Tanous         // default oem data
708*0f74e643SEd Tanous         nlohmann::json& oem = res.jsonValue["Oem"];
709*0f74e643SEd Tanous         nlohmann::json& oemOpenbmc = oem["OpenBmc"];
710*0f74e643SEd Tanous         oem["@odata.type"] = "#OemManager.Oem";
711*0f74e643SEd Tanous         oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem";
712*0f74e643SEd Tanous         oem["@odata.context"] = "/redfish/v1/$metadata#OemManager.Oem";
713*0f74e643SEd Tanous         oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc";
714*0f74e643SEd Tanous         oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc";
715*0f74e643SEd Tanous         oemOpenbmc["@odata.context"] =
716*0f74e643SEd Tanous             "/redfish/v1/$metadata#OemManager.OpenBmc";
717*0f74e643SEd Tanous 
718ed5befbdSJennifer Lee         // Update Actions object.
719*0f74e643SEd Tanous         nlohmann::json& manager_reset =
720*0f74e643SEd Tanous             res.jsonValue["Actions"]["#Manager.Reset"];
721ed5befbdSJennifer Lee         manager_reset["target"] =
722ed5befbdSJennifer Lee             "/redfish/v1/Managers/bmc/Actions/Manager.Reset";
723ed5befbdSJennifer Lee         manager_reset["ResetType@Redfish.AllowableValues"] = {
724ed5befbdSJennifer Lee             "GracefulRestart"};
725ca537928SJennifer Lee 
726*0f74e643SEd Tanous         res.jsonValue["DateTime"] = getDateTime();
727ed5befbdSJennifer Lee         std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res);
7285b4aa86bSJames Feist 
729ca537928SJennifer Lee         crow::connections::systemBus->async_method_call(
730ca537928SJennifer Lee             [asyncResp](const boost::system::error_code ec,
7315b4aa86bSJames Feist                         const dbus::utility::ManagedObjectType& resp) {
732ca537928SJennifer Lee                 if (ec)
733ca537928SJennifer Lee                 {
734ca537928SJennifer Lee                     BMCWEB_LOG_ERROR << "Error while getting Software Version";
735f12894f8SJason M. Bills                     messages::internalError(asyncResp->res);
736ca537928SJennifer Lee                     return;
737ca537928SJennifer Lee                 }
738ca537928SJennifer Lee 
739ca537928SJennifer Lee                 for (auto& objpath : resp)
740ca537928SJennifer Lee                 {
741ca537928SJennifer Lee                     for (auto& interface : objpath.second)
742ca537928SJennifer Lee                     {
743ca537928SJennifer Lee                         // If interface is xyz.openbmc_project.Software.Version,
744ca537928SJennifer Lee                         // this is what we're looking for.
745ca537928SJennifer Lee                         if (interface.first ==
746ca537928SJennifer Lee                             "xyz.openbmc_project.Software.Version")
747ca537928SJennifer Lee                         {
748ca537928SJennifer Lee                             // Cut out everyting until last "/", ...
749ca537928SJennifer Lee                             const std::string& iface_id = objpath.first;
750ca537928SJennifer Lee                             for (auto& property : interface.second)
751ca537928SJennifer Lee                             {
752ca537928SJennifer Lee                                 if (property.first == "Version")
753ca537928SJennifer Lee                                 {
754ca537928SJennifer Lee                                     const std::string* value =
755ca537928SJennifer Lee                                         mapbox::getPtr<const std::string>(
756ca537928SJennifer Lee                                             property.second);
757ca537928SJennifer Lee                                     if (value == nullptr)
758ca537928SJennifer Lee                                     {
759ca537928SJennifer Lee                                         continue;
760ca537928SJennifer Lee                                     }
761ca537928SJennifer Lee                                     asyncResp->res
762ca537928SJennifer Lee                                         .jsonValue["FirmwareVersion"] = *value;
763ca537928SJennifer Lee                                 }
764ca537928SJennifer Lee                             }
765ca537928SJennifer Lee                         }
766ca537928SJennifer Lee                     }
767ca537928SJennifer Lee                 }
768ca537928SJennifer Lee             },
769ca537928SJennifer Lee             "xyz.openbmc_project.Software.BMC.Updater",
770ca537928SJennifer Lee             "/xyz/openbmc_project/software",
771ca537928SJennifer Lee             "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
7725b4aa86bSJames Feist         getPidValues(asyncResp);
7735b4aa86bSJames Feist     }
77483ff9ab6SJames Feist     void setPidValues(std::shared_ptr<AsyncResp> response,
77583ff9ab6SJames Feist                       const nlohmann::json& data)
77683ff9ab6SJames Feist     {
77783ff9ab6SJames Feist         // todo(james): might make sense to do a mapper call here if this
77883ff9ab6SJames Feist         // interface gets more traction
77983ff9ab6SJames Feist         crow::connections::systemBus->async_method_call(
78083ff9ab6SJames Feist             [response,
78183ff9ab6SJames Feist              data](const boost::system::error_code ec,
78283ff9ab6SJames Feist                    const dbus::utility::ManagedObjectType& managedObj) {
78383ff9ab6SJames Feist                 if (ec)
78483ff9ab6SJames Feist                 {
78583ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Error communicating to Entity Manager";
78635a62c7cSJason M. Bills                     messages::internalError(response->res);
78783ff9ab6SJames Feist                     return;
78883ff9ab6SJames Feist                 }
78983ff9ab6SJames Feist                 for (const auto& type : data.items())
79083ff9ab6SJames Feist                 {
79183ff9ab6SJames Feist                     if (!type.value().is_object())
79283ff9ab6SJames Feist                     {
79383ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Illegal Type " << type.key();
79435a62c7cSJason M. Bills                         messages::propertyValueFormatError(
79535a62c7cSJason M. Bills                             response->res, type.value(), type.key());
79683ff9ab6SJames Feist                         return;
79783ff9ab6SJames Feist                     }
79883ff9ab6SJames Feist                     for (const auto& record : type.value().items())
79983ff9ab6SJames Feist                     {
80083ff9ab6SJames Feist                         const std::string& name = record.key();
80183ff9ab6SJames Feist                         auto pathItr =
80283ff9ab6SJames Feist                             std::find_if(managedObj.begin(), managedObj.end(),
80383ff9ab6SJames Feist                                          [&name](const auto& obj) {
80483ff9ab6SJames Feist                                              return boost::algorithm::ends_with(
80583ff9ab6SJames Feist                                                  obj.first.str, name);
80683ff9ab6SJames Feist                                          });
80783ff9ab6SJames Feist                         boost::container::flat_map<
80883ff9ab6SJames Feist                             std::string, dbus::utility::DbusVariantType>
80983ff9ab6SJames Feist                             output;
81083ff9ab6SJames Feist 
81183ff9ab6SJames Feist                         output.reserve(16); // The pid interface length
81283ff9ab6SJames Feist 
81383ff9ab6SJames Feist                         // determines if we're patching entity-manager or
81483ff9ab6SJames Feist                         // creating a new object
81583ff9ab6SJames Feist                         bool createNewObject = (pathItr == managedObj.end());
81683ff9ab6SJames Feist                         if (type.key() == "PidControllers" ||
81783ff9ab6SJames Feist                             type.key() == "FanControllers")
81883ff9ab6SJames Feist                         {
81983ff9ab6SJames Feist                             if (!createNewObject &&
82083ff9ab6SJames Feist                                 pathItr->second.find(pidConfigurationIface) ==
82183ff9ab6SJames Feist                                     pathItr->second.end())
82283ff9ab6SJames Feist                             {
82383ff9ab6SJames Feist                                 createNewObject = true;
82483ff9ab6SJames Feist                             }
82583ff9ab6SJames Feist                         }
82683ff9ab6SJames Feist                         else if (!createNewObject &&
82783ff9ab6SJames Feist                                  pathItr->second.find(
82883ff9ab6SJames Feist                                      pidZoneConfigurationIface) ==
82983ff9ab6SJames Feist                                      pathItr->second.end())
83083ff9ab6SJames Feist                         {
83183ff9ab6SJames Feist                             createNewObject = true;
83283ff9ab6SJames Feist                         }
83383ff9ab6SJames Feist                         output["Name"] =
83483ff9ab6SJames Feist                             boost::replace_all_copy(name, "_", " ");
83583ff9ab6SJames Feist 
83683ff9ab6SJames Feist                         std::string chassis;
83783ff9ab6SJames Feist                         CreatePIDRet ret = createPidInterface(
83883ff9ab6SJames Feist                             response, type.key(), record.value(),
83983ff9ab6SJames Feist                             pathItr->first.str, managedObj, createNewObject,
84083ff9ab6SJames Feist                             output, chassis);
84183ff9ab6SJames Feist                         if (ret == CreatePIDRet::fail)
84283ff9ab6SJames Feist                         {
84383ff9ab6SJames Feist                             return;
84483ff9ab6SJames Feist                         }
84583ff9ab6SJames Feist                         else if (ret == CreatePIDRet::del)
84683ff9ab6SJames Feist                         {
84783ff9ab6SJames Feist                             continue;
84883ff9ab6SJames Feist                         }
84983ff9ab6SJames Feist 
85083ff9ab6SJames Feist                         if (!createNewObject)
85183ff9ab6SJames Feist                         {
85283ff9ab6SJames Feist                             for (const auto& property : output)
85383ff9ab6SJames Feist                             {
85483ff9ab6SJames Feist                                 const char* iface =
85583ff9ab6SJames Feist                                     type.key() == "FanZones"
85683ff9ab6SJames Feist                                         ? pidZoneConfigurationIface
85783ff9ab6SJames Feist                                         : pidConfigurationIface;
85883ff9ab6SJames Feist                                 crow::connections::systemBus->async_method_call(
85983ff9ab6SJames Feist                                     [response,
86083ff9ab6SJames Feist                                      propertyName{std::string(property.first)}](
86183ff9ab6SJames Feist                                         const boost::system::error_code ec) {
86283ff9ab6SJames Feist                                         if (ec)
86383ff9ab6SJames Feist                                         {
86483ff9ab6SJames Feist                                             BMCWEB_LOG_ERROR
86583ff9ab6SJames Feist                                                 << "Error patching "
86683ff9ab6SJames Feist                                                 << propertyName << ": " << ec;
86735a62c7cSJason M. Bills                                             messages::internalError(
86835a62c7cSJason M. Bills                                                 response->res);
86983ff9ab6SJames Feist                                         }
87083ff9ab6SJames Feist                                     },
87183ff9ab6SJames Feist                                     "xyz.openbmc_project.EntityManager",
87283ff9ab6SJames Feist                                     pathItr->first.str,
87383ff9ab6SJames Feist                                     "org.freedesktop.DBus.Properties", "Set",
87483ff9ab6SJames Feist                                     std::string(iface), property.first,
87583ff9ab6SJames Feist                                     property.second);
87683ff9ab6SJames Feist                             }
87783ff9ab6SJames Feist                         }
87883ff9ab6SJames Feist                         else
87983ff9ab6SJames Feist                         {
88083ff9ab6SJames Feist                             if (chassis.empty())
88183ff9ab6SJames Feist                             {
88283ff9ab6SJames Feist                                 BMCWEB_LOG_ERROR
88383ff9ab6SJames Feist                                     << "Failed to get chassis from config";
88435a62c7cSJason M. Bills                                 messages::invalidObject(response->res, name);
88583ff9ab6SJames Feist                                 return;
88683ff9ab6SJames Feist                             }
88783ff9ab6SJames Feist 
88883ff9ab6SJames Feist                             bool foundChassis = false;
88983ff9ab6SJames Feist                             for (const auto& obj : managedObj)
89083ff9ab6SJames Feist                             {
89183ff9ab6SJames Feist                                 if (boost::algorithm::ends_with(obj.first.str,
89283ff9ab6SJames Feist                                                                 chassis))
89383ff9ab6SJames Feist                                 {
89483ff9ab6SJames Feist                                     chassis = obj.first.str;
89583ff9ab6SJames Feist                                     foundChassis = true;
89683ff9ab6SJames Feist                                     break;
89783ff9ab6SJames Feist                                 }
89883ff9ab6SJames Feist                             }
89983ff9ab6SJames Feist                             if (!foundChassis)
90083ff9ab6SJames Feist                             {
90183ff9ab6SJames Feist                                 BMCWEB_LOG_ERROR
90283ff9ab6SJames Feist                                     << "Failed to find chassis on dbus";
90383ff9ab6SJames Feist                                 messages::resourceMissingAtURI(
90435a62c7cSJason M. Bills                                     response->res,
90535a62c7cSJason M. Bills                                     "/redfish/v1/Chassis/" + chassis);
90683ff9ab6SJames Feist                                 return;
90783ff9ab6SJames Feist                             }
90883ff9ab6SJames Feist 
90983ff9ab6SJames Feist                             crow::connections::systemBus->async_method_call(
91083ff9ab6SJames Feist                                 [response](const boost::system::error_code ec) {
91183ff9ab6SJames Feist                                     if (ec)
91283ff9ab6SJames Feist                                     {
91383ff9ab6SJames Feist                                         BMCWEB_LOG_ERROR
91483ff9ab6SJames Feist                                             << "Error Adding Pid Object " << ec;
91535a62c7cSJason M. Bills                                         messages::internalError(response->res);
91683ff9ab6SJames Feist                                     }
91783ff9ab6SJames Feist                                 },
91883ff9ab6SJames Feist                                 "xyz.openbmc_project.EntityManager", chassis,
91983ff9ab6SJames Feist                                 "xyz.openbmc_project.AddObject", "AddObject",
92083ff9ab6SJames Feist                                 output);
92183ff9ab6SJames Feist                         }
92283ff9ab6SJames Feist                     }
92383ff9ab6SJames Feist                 }
92483ff9ab6SJames Feist             },
92583ff9ab6SJames Feist             "xyz.openbmc_project.EntityManager", "/", objectManagerIface,
92683ff9ab6SJames Feist             "GetManagedObjects");
92783ff9ab6SJames Feist     }
9285b4aa86bSJames Feist 
9295b4aa86bSJames Feist     void doPatch(crow::Response& res, const crow::Request& req,
9305b4aa86bSJames Feist                  const std::vector<std::string>& params) override
9315b4aa86bSJames Feist     {
93283ff9ab6SJames Feist         nlohmann::json patch;
93383ff9ab6SJames Feist         if (!json_util::processJsonFromRequest(res, req, patch))
93483ff9ab6SJames Feist         {
93583ff9ab6SJames Feist             return;
93683ff9ab6SJames Feist         }
93783ff9ab6SJames Feist         std::shared_ptr<AsyncResp> response = std::make_shared<AsyncResp>(res);
93883ff9ab6SJames Feist         for (const auto& topLevel : patch.items())
93983ff9ab6SJames Feist         {
94083ff9ab6SJames Feist             if (topLevel.key() == "Oem")
94183ff9ab6SJames Feist             {
94283ff9ab6SJames Feist                 if (!topLevel.value().is_object())
94383ff9ab6SJames Feist                 {
94483ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Bad Patch " << topLevel.key();
94535a62c7cSJason M. Bills                     messages::propertyValueFormatError(
94635a62c7cSJason M. Bills                         response->res, topLevel.key(), "OemManager.Oem");
94783ff9ab6SJames Feist                     return;
94883ff9ab6SJames Feist                 }
94983ff9ab6SJames Feist             }
95083ff9ab6SJames Feist             else
95183ff9ab6SJames Feist             {
95283ff9ab6SJames Feist                 BMCWEB_LOG_ERROR << "Bad Patch " << topLevel.key();
95335a62c7cSJason M. Bills                 messages::propertyUnknown(response->res, topLevel.key());
95483ff9ab6SJames Feist                 return;
95583ff9ab6SJames Feist             }
95683ff9ab6SJames Feist             for (const auto& oemLevel : topLevel.value().items())
95783ff9ab6SJames Feist             {
95883ff9ab6SJames Feist                 if (oemLevel.key() == "OpenBmc")
95983ff9ab6SJames Feist                 {
96083ff9ab6SJames Feist                     if (!oemLevel.value().is_object())
96183ff9ab6SJames Feist                     {
96283ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Bad Patch " << oemLevel.key();
96335a62c7cSJason M. Bills                         messages::propertyValueFormatError(
96435a62c7cSJason M. Bills                             response->res, topLevel.key(),
96535a62c7cSJason M. Bills                             "OemManager.OpenBmc");
96683ff9ab6SJames Feist                         return;
96783ff9ab6SJames Feist                     }
96883ff9ab6SJames Feist                     for (const auto& typeLevel : oemLevel.value().items())
96983ff9ab6SJames Feist                     {
97083ff9ab6SJames Feist 
97183ff9ab6SJames Feist                         if (typeLevel.key() == "Fan")
97283ff9ab6SJames Feist                         {
97383ff9ab6SJames Feist                             if (!typeLevel.value().is_object())
97483ff9ab6SJames Feist                             {
97583ff9ab6SJames Feist                                 BMCWEB_LOG_ERROR << "Bad Patch "
97683ff9ab6SJames Feist                                                  << typeLevel.key();
97783ff9ab6SJames Feist                                 messages::propertyValueFormatError(
97835a62c7cSJason M. Bills                                     response->res, typeLevel.value().dump(),
97935a62c7cSJason M. Bills                                     typeLevel.key());
98083ff9ab6SJames Feist                                 return;
98183ff9ab6SJames Feist                             }
98283ff9ab6SJames Feist                             setPidValues(response,
98383ff9ab6SJames Feist                                          std::move(typeLevel.value()));
98483ff9ab6SJames Feist                         }
98583ff9ab6SJames Feist                         else
98683ff9ab6SJames Feist                         {
98783ff9ab6SJames Feist                             BMCWEB_LOG_ERROR << "Bad Patch " << typeLevel.key();
98835a62c7cSJason M. Bills                             messages::propertyUnknown(response->res,
98935a62c7cSJason M. Bills                                                       typeLevel.key());
99083ff9ab6SJames Feist                             return;
99183ff9ab6SJames Feist                         }
99283ff9ab6SJames Feist                     }
99383ff9ab6SJames Feist                 }
99483ff9ab6SJames Feist                 else
99583ff9ab6SJames Feist                 {
99683ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Bad Patch " << oemLevel.key();
99735a62c7cSJason M. Bills                     messages::propertyUnknown(response->res, oemLevel.key());
99883ff9ab6SJames Feist                     return;
99983ff9ab6SJames Feist                 }
100083ff9ab6SJames Feist             }
100183ff9ab6SJames Feist         }
10029c310685SBorawski.Lukasz     }
10039c310685SBorawski.Lukasz 
10041abe55efSEd Tanous     std::string getDateTime() const
10051abe55efSEd Tanous     {
10069c310685SBorawski.Lukasz         std::array<char, 128> dateTime;
10079c310685SBorawski.Lukasz         std::string redfishDateTime("0000-00-00T00:00:00Z00:00");
10089c310685SBorawski.Lukasz         std::time_t time = std::time(nullptr);
10099c310685SBorawski.Lukasz 
10109c310685SBorawski.Lukasz         if (std::strftime(dateTime.begin(), dateTime.size(), "%FT%T%z",
10111abe55efSEd Tanous                           std::localtime(&time)))
10121abe55efSEd Tanous         {
10139c310685SBorawski.Lukasz             // insert the colon required by the ISO 8601 standard
10149c310685SBorawski.Lukasz             redfishDateTime = std::string(dateTime.data());
10159c310685SBorawski.Lukasz             redfishDateTime.insert(redfishDateTime.end() - 2, ':');
10169c310685SBorawski.Lukasz         }
10179c310685SBorawski.Lukasz 
10189c310685SBorawski.Lukasz         return redfishDateTime;
10199c310685SBorawski.Lukasz     }
1020*0f74e643SEd Tanous 
1021*0f74e643SEd Tanous     std::string uuid;
10229c310685SBorawski.Lukasz };
10239c310685SBorawski.Lukasz 
10241abe55efSEd Tanous class ManagerCollection : public Node
10251abe55efSEd Tanous {
10269c310685SBorawski.Lukasz   public:
10271abe55efSEd Tanous     ManagerCollection(CrowApp& app) : Node(app, "/redfish/v1/Managers/")
10281abe55efSEd Tanous     {
1029a434f2bdSEd Tanous         entityPrivileges = {
1030a434f2bdSEd Tanous             {boost::beast::http::verb::get, {{"Login"}}},
1031e0d918bcSEd Tanous             {boost::beast::http::verb::head, {{"Login"}}},
1032e0d918bcSEd Tanous             {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
1033e0d918bcSEd Tanous             {boost::beast::http::verb::put, {{"ConfigureManager"}}},
1034e0d918bcSEd Tanous             {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
1035e0d918bcSEd Tanous             {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
10369c310685SBorawski.Lukasz     }
10379c310685SBorawski.Lukasz 
10389c310685SBorawski.Lukasz   private:
103955c7b7a2SEd Tanous     void doGet(crow::Response& res, const crow::Request& req,
10401abe55efSEd Tanous                const std::vector<std::string>& params) override
10411abe55efSEd Tanous     {
104283ff9ab6SJames Feist         // Collections don't include the static data added by SubRoute
104383ff9ab6SJames Feist         // because it has a duplicate entry for members
104455c7b7a2SEd Tanous         res.jsonValue["@odata.id"] = "/redfish/v1/Managers";
104555c7b7a2SEd Tanous         res.jsonValue["@odata.type"] = "#ManagerCollection.ManagerCollection";
104655c7b7a2SEd Tanous         res.jsonValue["@odata.context"] =
104755c7b7a2SEd Tanous             "/redfish/v1/$metadata#ManagerCollection.ManagerCollection";
104855c7b7a2SEd Tanous         res.jsonValue["Name"] = "Manager Collection";
104955c7b7a2SEd Tanous         res.jsonValue["Members@odata.count"] = 1;
105055c7b7a2SEd Tanous         res.jsonValue["Members"] = {
10515b4aa86bSJames Feist             {{"@odata.id", "/redfish/v1/Managers/bmc"}}};
10529c310685SBorawski.Lukasz         res.end();
10539c310685SBorawski.Lukasz     }
10549c310685SBorawski.Lukasz };
10559c310685SBorawski.Lukasz } // namespace redfish
1056