xref: /openbmc/bmcweb/features/redfish/lib/managers.hpp (revision 7517658e75c07ba0ac0e9641fd5976175675f77f)
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 =
1771b6b96c5SEd Tanous                         sdbusplus::message::variant_ns::get_if<std::string>(
1781b6b96c5SEd Tanous                             &findName->second);
1795b4aa86bSJames Feist                     if (namePtr == nullptr)
1805b4aa86bSJames Feist                     {
1815b4aa86bSJames Feist                         BMCWEB_LOG_ERROR << "Pid Name Field illegal";
1825b4aa86bSJames Feist                         return;
1835b4aa86bSJames Feist                     }
1845b4aa86bSJames Feist 
1855b4aa86bSJames Feist                     std::string name = *namePtr;
1865b4aa86bSJames Feist                     dbus::utility::escapePathForDbus(name);
1875b4aa86bSJames Feist                     if (intfPair.first == pidZoneConfigurationIface)
1885b4aa86bSJames Feist                     {
1895b4aa86bSJames Feist                         std::string chassis;
1905b4aa86bSJames Feist                         if (!dbus::utility::getNthStringFromPath(
1915b4aa86bSJames Feist                                 pathPair.first.str, 5, chassis))
1925b4aa86bSJames Feist                         {
1935b4aa86bSJames Feist                             chassis = "#IllegalValue";
1945b4aa86bSJames Feist                         }
1955b4aa86bSJames Feist                         nlohmann::json& zone = zones[name];
1965b4aa86bSJames Feist                         zone["Chassis"] = {
1975b4aa86bSJames Feist                             {"@odata.id", "/redfish/v1/Chassis/" + chassis}};
1985b4aa86bSJames Feist                         zone["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/"
1995b4aa86bSJames Feist                                             "OpenBmc/Fan/FanZones/" +
2005b4aa86bSJames Feist                                             name;
2015b4aa86bSJames Feist                         zone["@odata.type"] = "#OemManager.FanZone";
2025b4aa86bSJames Feist                         zone["@odata.context"] =
2035b4aa86bSJames Feist                             "/redfish/v1/$metadata#OemManager.FanZone";
2045b4aa86bSJames Feist                     }
2055b4aa86bSJames Feist 
2065b4aa86bSJames Feist                     for (const auto& propertyPair : intfPair.second)
2075b4aa86bSJames Feist                     {
2085b4aa86bSJames Feist                         if (propertyPair.first == "Type" ||
2095b4aa86bSJames Feist                             propertyPair.first == "Class" ||
2105b4aa86bSJames Feist                             propertyPair.first == "Name")
2115b4aa86bSJames Feist                         {
2125b4aa86bSJames Feist                             continue;
2135b4aa86bSJames Feist                         }
2145b4aa86bSJames Feist 
2155b4aa86bSJames Feist                         // zones
2165b4aa86bSJames Feist                         if (intfPair.first == pidZoneConfigurationIface)
2175b4aa86bSJames Feist                         {
2181b6b96c5SEd Tanous                             const double* ptr =
2191b6b96c5SEd Tanous                                 sdbusplus::message::variant_ns::get_if<double>(
2201b6b96c5SEd Tanous                                     &propertyPair.second);
2215b4aa86bSJames Feist                             if (ptr == nullptr)
2225b4aa86bSJames Feist                             {
2235b4aa86bSJames Feist                                 BMCWEB_LOG_ERROR << "Field Illegal "
2245b4aa86bSJames Feist                                                  << propertyPair.first;
225f12894f8SJason M. Bills                                 messages::internalError(asyncResp->res);
2265b4aa86bSJames Feist                                 return;
2275b4aa86bSJames Feist                             }
2285b4aa86bSJames Feist                             zones[name][propertyPair.first] = *ptr;
2295b4aa86bSJames Feist                         }
2305b4aa86bSJames Feist 
2315b4aa86bSJames Feist                         // pid and fans are off the same configuration
2325b4aa86bSJames Feist                         if (intfPair.first == pidConfigurationIface)
2335b4aa86bSJames Feist                         {
2345b4aa86bSJames Feist                             const std::string* classPtr = nullptr;
2355b4aa86bSJames Feist                             auto findClass = intfPair.second.find("Class");
2365b4aa86bSJames Feist                             if (findClass != intfPair.second.end())
2375b4aa86bSJames Feist                             {
2381b6b96c5SEd Tanous                                 classPtr =
2391b6b96c5SEd Tanous                                     sdbusplus::message::variant_ns::get_if<
2401b6b96c5SEd Tanous                                         std::string>(&findClass->second);
2415b4aa86bSJames Feist                             }
2425b4aa86bSJames Feist                             if (classPtr == nullptr)
2435b4aa86bSJames Feist                             {
2445b4aa86bSJames Feist                                 BMCWEB_LOG_ERROR << "Pid Class Field illegal";
245a08b46ccSJason M. Bills                                 messages::internalError(asyncResp->res);
2465b4aa86bSJames Feist                                 return;
2475b4aa86bSJames Feist                             }
2485b4aa86bSJames Feist                             bool isFan = *classPtr == "fan";
2495b4aa86bSJames Feist                             nlohmann::json& element =
2505b4aa86bSJames Feist                                 isFan ? fans[name] : pids[name];
2515b4aa86bSJames Feist                             if (isFan)
2525b4aa86bSJames Feist                             {
2535b4aa86bSJames Feist                                 element["@odata.id"] =
2545b4aa86bSJames Feist                                     "/redfish/v1/Managers/bmc#/Oem/"
2555b4aa86bSJames Feist                                     "OpenBmc/Fan/FanControllers/" +
2565b4aa86bSJames Feist                                     std::string(name);
2575b4aa86bSJames Feist                                 element["@odata.type"] =
2585b4aa86bSJames Feist                                     "#OemManager.FanController";
2595b4aa86bSJames Feist 
2605b4aa86bSJames Feist                                 element["@odata.context"] =
2615b4aa86bSJames Feist                                     "/redfish/v1/"
2625b4aa86bSJames Feist                                     "$metadata#OemManager.FanController";
2635b4aa86bSJames Feist                             }
2645b4aa86bSJames Feist                             else
2655b4aa86bSJames Feist                             {
2665b4aa86bSJames Feist                                 element["@odata.id"] =
2675b4aa86bSJames Feist                                     "/redfish/v1/Managers/bmc#/Oem/"
2685b4aa86bSJames Feist                                     "OpenBmc/Fan/PidControllers/" +
2695b4aa86bSJames Feist                                     std::string(name);
2705b4aa86bSJames Feist                                 element["@odata.type"] =
2715b4aa86bSJames Feist                                     "#OemManager.PidController";
2725b4aa86bSJames Feist                                 element["@odata.context"] =
2735b4aa86bSJames Feist                                     "/redfish/v1/$metadata"
2745b4aa86bSJames Feist                                     "#OemManager.PidController";
2755b4aa86bSJames Feist                             }
2765b4aa86bSJames Feist 
2775b4aa86bSJames Feist                             if (propertyPair.first == "Zones")
2785b4aa86bSJames Feist                             {
2795b4aa86bSJames Feist                                 const std::vector<std::string>* inputs =
2801b6b96c5SEd Tanous                                     sdbusplus::message::variant_ns::get_if<
2811b6b96c5SEd Tanous                                         std::vector<std::string>>(
2821b6b96c5SEd Tanous                                         &propertyPair.second);
2835b4aa86bSJames Feist 
2845b4aa86bSJames Feist                                 if (inputs == nullptr)
2855b4aa86bSJames Feist                                 {
2865b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR
2875b4aa86bSJames Feist                                         << "Zones Pid Field Illegal";
288a08b46ccSJason M. Bills                                     messages::internalError(asyncResp->res);
2895b4aa86bSJames Feist                                     return;
2905b4aa86bSJames Feist                                 }
2915b4aa86bSJames Feist                                 auto& data = element[propertyPair.first];
2925b4aa86bSJames Feist                                 data = nlohmann::json::array();
2935b4aa86bSJames Feist                                 for (std::string itemCopy : *inputs)
2945b4aa86bSJames Feist                                 {
2955b4aa86bSJames Feist                                     dbus::utility::escapePathForDbus(itemCopy);
2965b4aa86bSJames Feist                                     data.push_back(
2975b4aa86bSJames Feist                                         {{"@odata.id",
2985b4aa86bSJames Feist                                           "/redfish/v1/Managers/bmc#/Oem/"
2995b4aa86bSJames Feist                                           "OpenBmc/Fan/FanZones/" +
3005b4aa86bSJames Feist                                               itemCopy}});
3015b4aa86bSJames Feist                                 }
3025b4aa86bSJames Feist                             }
3035b4aa86bSJames Feist                             // todo(james): may never happen, but this
3045b4aa86bSJames Feist                             // assumes configuration data referenced in the
3055b4aa86bSJames Feist                             // PID config is provided by the same daemon, we
3065b4aa86bSJames Feist                             // could add another loop to cover all cases,
3075b4aa86bSJames Feist                             // but I'm okay kicking this can down the road a
3085b4aa86bSJames Feist                             // bit
3095b4aa86bSJames Feist 
3105b4aa86bSJames Feist                             else if (propertyPair.first == "Inputs" ||
3115b4aa86bSJames Feist                                      propertyPair.first == "Outputs")
3125b4aa86bSJames Feist                             {
3135b4aa86bSJames Feist                                 auto& data = element[propertyPair.first];
3145b4aa86bSJames Feist                                 const std::vector<std::string>* inputs =
3151b6b96c5SEd Tanous                                     sdbusplus::message::variant_ns::get_if<
3161b6b96c5SEd Tanous                                         std::vector<std::string>>(
3171b6b96c5SEd Tanous                                         &propertyPair.second);
3185b4aa86bSJames Feist 
3195b4aa86bSJames Feist                                 if (inputs == nullptr)
3205b4aa86bSJames Feist                                 {
3215b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
3225b4aa86bSJames Feist                                                      << propertyPair.first;
323f12894f8SJason M. Bills                                     messages::internalError(asyncResp->res);
3245b4aa86bSJames Feist                                     return;
3255b4aa86bSJames Feist                                 }
3265b4aa86bSJames Feist                                 data = *inputs;
3275b4aa86bSJames Feist                             } // doubles
3285b4aa86bSJames Feist                             else if (propertyPair.first ==
3295b4aa86bSJames Feist                                          "FFGainCoefficient" ||
3305b4aa86bSJames Feist                                      propertyPair.first == "FFOffCoefficient" ||
3315b4aa86bSJames Feist                                      propertyPair.first == "ICoefficient" ||
3325b4aa86bSJames Feist                                      propertyPair.first == "ILimitMax" ||
3335b4aa86bSJames Feist                                      propertyPair.first == "ILimitMin" ||
3345b4aa86bSJames Feist                                      propertyPair.first == "OutLimitMax" ||
3355b4aa86bSJames Feist                                      propertyPair.first == "OutLimitMin" ||
3365b4aa86bSJames Feist                                      propertyPair.first == "PCoefficient" ||
3375b4aa86bSJames Feist                                      propertyPair.first == "SlewNeg" ||
3385b4aa86bSJames Feist                                      propertyPair.first == "SlewPos")
3395b4aa86bSJames Feist                             {
3405b4aa86bSJames Feist                                 const double* ptr =
3411b6b96c5SEd Tanous                                     sdbusplus::message::variant_ns::get_if<
3421b6b96c5SEd Tanous                                         double>(&propertyPair.second);
3435b4aa86bSJames Feist                                 if (ptr == nullptr)
3445b4aa86bSJames Feist                                 {
3455b4aa86bSJames Feist                                     BMCWEB_LOG_ERROR << "Field Illegal "
3465b4aa86bSJames Feist                                                      << propertyPair.first;
347f12894f8SJason M. Bills                                     messages::internalError(asyncResp->res);
3485b4aa86bSJames Feist                                     return;
3495b4aa86bSJames Feist                                 }
3505b4aa86bSJames Feist                                 element[propertyPair.first] = *ptr;
3515b4aa86bSJames Feist                             }
3525b4aa86bSJames Feist                         }
3535b4aa86bSJames Feist                     }
3545b4aa86bSJames Feist                 }
3555b4aa86bSJames Feist             }
3565b4aa86bSJames Feist         },
3575b4aa86bSJames Feist         connection, path, objectManagerIface, "GetManagedObjects");
3585b4aa86bSJames Feist }
359ca537928SJennifer Lee 
36083ff9ab6SJames Feist enum class CreatePIDRet
36183ff9ab6SJames Feist {
36283ff9ab6SJames Feist     fail,
36383ff9ab6SJames Feist     del,
36483ff9ab6SJames Feist     patch
36583ff9ab6SJames Feist };
36683ff9ab6SJames Feist 
36783ff9ab6SJames Feist static CreatePIDRet createPidInterface(
36883ff9ab6SJames Feist     const std::shared_ptr<AsyncResp>& response, const std::string& type,
36983ff9ab6SJames Feist     const nlohmann::json& record, const std::string& path,
37083ff9ab6SJames Feist     const dbus::utility::ManagedObjectType& managedObj, bool createNewObject,
37183ff9ab6SJames Feist     boost::container::flat_map<std::string, dbus::utility::DbusVariantType>&
37283ff9ab6SJames Feist         output,
37383ff9ab6SJames Feist     std::string& chassis)
37483ff9ab6SJames Feist {
37583ff9ab6SJames Feist 
37683ff9ab6SJames Feist     if (type == "PidControllers" || type == "FanControllers")
37783ff9ab6SJames Feist     {
37883ff9ab6SJames Feist         if (createNewObject)
37983ff9ab6SJames Feist         {
38083ff9ab6SJames Feist             output["Class"] = type == "PidControllers" ? std::string("temp")
38183ff9ab6SJames Feist                                                        : std::string("fan");
38283ff9ab6SJames Feist             output["Type"] = std::string("Pid");
38383ff9ab6SJames Feist         }
38483ff9ab6SJames Feist         else if (record == nullptr)
38583ff9ab6SJames Feist         {
38683ff9ab6SJames Feist             // delete interface
38783ff9ab6SJames Feist             crow::connections::systemBus->async_method_call(
38883ff9ab6SJames Feist                 [response,
38983ff9ab6SJames Feist                  path{std::string(path)}](const boost::system::error_code ec) {
39083ff9ab6SJames Feist                     if (ec)
39183ff9ab6SJames Feist                     {
39283ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Error patching " << path << ": "
39383ff9ab6SJames Feist                                          << ec;
39435a62c7cSJason M. Bills                         messages::internalError(response->res);
39583ff9ab6SJames Feist                     }
39683ff9ab6SJames Feist                 },
39783ff9ab6SJames Feist                 "xyz.openbmc_project.EntityManager", path,
39883ff9ab6SJames Feist                 pidConfigurationIface, "Delete");
39983ff9ab6SJames Feist             return CreatePIDRet::del;
40083ff9ab6SJames Feist         }
40183ff9ab6SJames Feist 
40283ff9ab6SJames Feist         for (auto& field : record.items())
40383ff9ab6SJames Feist         {
40483ff9ab6SJames Feist             if (field.key() == "Zones")
40583ff9ab6SJames Feist             {
40683ff9ab6SJames Feist                 if (!field.value().is_array())
40783ff9ab6SJames Feist                 {
40883ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
40935a62c7cSJason M. Bills                     messages::propertyValueFormatError(
41035a62c7cSJason M. Bills                         response->res, field.value(), field.key());
41183ff9ab6SJames Feist                     return CreatePIDRet::fail;
41283ff9ab6SJames Feist                 }
41383ff9ab6SJames Feist                 std::vector<std::string> inputs;
41483ff9ab6SJames Feist                 for (const auto& odata : field.value().items())
41583ff9ab6SJames Feist                 {
41683ff9ab6SJames Feist                     for (const auto& value : odata.value().items())
41783ff9ab6SJames Feist                     {
41883ff9ab6SJames Feist                         const std::string* path =
41983ff9ab6SJames Feist                             value.value().get_ptr<const std::string*>();
42083ff9ab6SJames Feist                         if (path == nullptr)
42183ff9ab6SJames Feist                         {
42283ff9ab6SJames Feist                             BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
42383ff9ab6SJames Feist                             messages::propertyValueFormatError(
42435a62c7cSJason M. Bills                                 response->res, field.value().dump(),
42535a62c7cSJason M. Bills                                 field.key());
42683ff9ab6SJames Feist                             return CreatePIDRet::fail;
42783ff9ab6SJames Feist                         }
42883ff9ab6SJames Feist                         std::string input;
42983ff9ab6SJames Feist                         if (!dbus::utility::getNthStringFromPath(*path, 4,
43083ff9ab6SJames Feist                                                                  input))
43183ff9ab6SJames Feist                         {
43283ff9ab6SJames Feist                             BMCWEB_LOG_ERROR << "Got invalid path " << *path;
43383ff9ab6SJames Feist                             messages::propertyValueFormatError(
43435a62c7cSJason M. Bills                                 response->res, field.value().dump(),
43535a62c7cSJason M. Bills                                 field.key());
43683ff9ab6SJames Feist                             return CreatePIDRet::fail;
43783ff9ab6SJames Feist                         }
43883ff9ab6SJames Feist                         boost::replace_all(input, "_", " ");
43983ff9ab6SJames Feist                         inputs.emplace_back(std::move(input));
44083ff9ab6SJames Feist                     }
44183ff9ab6SJames Feist                 }
44283ff9ab6SJames Feist                 output["Zones"] = std::move(inputs);
44383ff9ab6SJames Feist             }
44483ff9ab6SJames Feist             else if (field.key() == "Inputs" || field.key() == "Outputs")
44583ff9ab6SJames Feist             {
44683ff9ab6SJames Feist                 if (!field.value().is_array())
44783ff9ab6SJames Feist                 {
44883ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
44935a62c7cSJason M. Bills                     messages::propertyValueFormatError(
45035a62c7cSJason M. Bills                         response->res, field.value().dump(), field.key());
45183ff9ab6SJames Feist                     return CreatePIDRet::fail;
45283ff9ab6SJames Feist                 }
45383ff9ab6SJames Feist                 std::vector<std::string> inputs;
45483ff9ab6SJames Feist                 for (const auto& value : field.value().items())
45583ff9ab6SJames Feist                 {
45683ff9ab6SJames Feist                     const std::string* sensor =
45783ff9ab6SJames Feist                         value.value().get_ptr<const std::string*>();
45883ff9ab6SJames Feist 
45983ff9ab6SJames Feist                     if (sensor == nullptr)
46083ff9ab6SJames Feist                     {
46183ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Illegal Type "
46283ff9ab6SJames Feist                                          << field.value().dump();
46383ff9ab6SJames Feist                         messages::propertyValueFormatError(
46435a62c7cSJason M. Bills                             response->res, field.value().dump(), field.key());
46583ff9ab6SJames Feist                         return CreatePIDRet::fail;
46683ff9ab6SJames Feist                     }
46783ff9ab6SJames Feist 
46883ff9ab6SJames Feist                     std::string input =
46983ff9ab6SJames Feist                         boost::replace_all_copy(*sensor, "_", " ");
47083ff9ab6SJames Feist                     inputs.push_back(std::move(input));
47183ff9ab6SJames Feist                     // try to find the sensor in the
47283ff9ab6SJames Feist                     // configuration
47383ff9ab6SJames Feist                     if (chassis.empty())
47483ff9ab6SJames Feist                     {
47583ff9ab6SJames Feist                         std::find_if(
47683ff9ab6SJames Feist                             managedObj.begin(), managedObj.end(),
47783ff9ab6SJames Feist                             [&chassis, sensor](const auto& obj) {
47883ff9ab6SJames Feist                                 if (boost::algorithm::ends_with(obj.first.str,
47983ff9ab6SJames Feist                                                                 *sensor))
48083ff9ab6SJames Feist                                 {
48183ff9ab6SJames Feist                                     return dbus::utility::getNthStringFromPath(
48283ff9ab6SJames Feist                                         obj.first.str, 5, chassis);
48383ff9ab6SJames Feist                                 }
48483ff9ab6SJames Feist                                 return false;
48583ff9ab6SJames Feist                             });
48683ff9ab6SJames Feist                     }
48783ff9ab6SJames Feist                 }
48883ff9ab6SJames Feist                 output[field.key()] = inputs;
48983ff9ab6SJames Feist             }
49083ff9ab6SJames Feist 
49183ff9ab6SJames Feist             // doubles
49283ff9ab6SJames Feist             else if (field.key() == "FFGainCoefficient" ||
49383ff9ab6SJames Feist                      field.key() == "FFOffCoefficient" ||
49483ff9ab6SJames Feist                      field.key() == "ICoefficient" ||
49583ff9ab6SJames Feist                      field.key() == "ILimitMax" || field.key() == "ILimitMin" ||
49683ff9ab6SJames Feist                      field.key() == "OutLimitMax" ||
49783ff9ab6SJames Feist                      field.key() == "OutLimitMin" ||
49883ff9ab6SJames Feist                      field.key() == "PCoefficient" ||
49983ff9ab6SJames Feist                      field.key() == "SetPoint" || field.key() == "SlewNeg" ||
50083ff9ab6SJames Feist                      field.key() == "SlewPos")
50183ff9ab6SJames Feist             {
50283ff9ab6SJames Feist                 const double* ptr = field.value().get_ptr<const double*>();
50383ff9ab6SJames Feist                 if (ptr == nullptr)
50483ff9ab6SJames Feist                 {
50583ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
50635a62c7cSJason M. Bills                     messages::propertyValueFormatError(
50735a62c7cSJason M. Bills                         response->res, field.value().dump(), field.key());
50883ff9ab6SJames Feist                     return CreatePIDRet::fail;
50983ff9ab6SJames Feist                 }
51083ff9ab6SJames Feist                 output[field.key()] = *ptr;
51183ff9ab6SJames Feist             }
51283ff9ab6SJames Feist 
51383ff9ab6SJames Feist             else
51483ff9ab6SJames Feist             {
51583ff9ab6SJames Feist                 BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
51635a62c7cSJason M. Bills                 messages::propertyUnknown(response->res, field.key());
51783ff9ab6SJames Feist                 return CreatePIDRet::fail;
51883ff9ab6SJames Feist             }
51983ff9ab6SJames Feist         }
52083ff9ab6SJames Feist     }
52183ff9ab6SJames Feist     else if (type == "FanZones")
52283ff9ab6SJames Feist     {
52383ff9ab6SJames Feist         if (!createNewObject && record == nullptr)
52483ff9ab6SJames Feist         {
52583ff9ab6SJames Feist             // delete interface
52683ff9ab6SJames Feist             crow::connections::systemBus->async_method_call(
52783ff9ab6SJames Feist                 [response,
52883ff9ab6SJames Feist                  path{std::string(path)}](const boost::system::error_code ec) {
52983ff9ab6SJames Feist                     if (ec)
53083ff9ab6SJames Feist                     {
53183ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Error patching " << path << ": "
53283ff9ab6SJames Feist                                          << ec;
53335a62c7cSJason M. Bills                         messages::internalError(response->res);
53483ff9ab6SJames Feist                     }
53583ff9ab6SJames Feist                 },
53683ff9ab6SJames Feist                 "xyz.openbmc_project.EntityManager", path,
53783ff9ab6SJames Feist                 pidZoneConfigurationIface, "Delete");
53883ff9ab6SJames Feist             return CreatePIDRet::del;
53983ff9ab6SJames Feist         }
54083ff9ab6SJames Feist         output["Type"] = std::string("Pid.Zone");
54183ff9ab6SJames Feist 
54283ff9ab6SJames Feist         for (auto& field : record.items())
54383ff9ab6SJames Feist         {
54483ff9ab6SJames Feist             if (field.key() == "Chassis")
54583ff9ab6SJames Feist             {
54683ff9ab6SJames Feist                 const std::string* chassisId = nullptr;
54783ff9ab6SJames Feist                 for (const auto& id : field.value().items())
54883ff9ab6SJames Feist                 {
54983ff9ab6SJames Feist                     if (id.key() != "@odata.id")
55083ff9ab6SJames Feist                     {
55183ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Illegal Type " << id.key();
55235a62c7cSJason M. Bills                         messages::propertyUnknown(response->res, field.key());
55383ff9ab6SJames Feist                         return CreatePIDRet::fail;
55483ff9ab6SJames Feist                     }
55583ff9ab6SJames Feist                     chassisId = id.value().get_ptr<const std::string*>();
55683ff9ab6SJames Feist                     if (chassisId == nullptr)
55783ff9ab6SJames Feist                     {
55883ff9ab6SJames Feist                         messages::createFailedMissingReqProperties(
55935a62c7cSJason M. Bills                             response->res, field.key());
56083ff9ab6SJames Feist                         return CreatePIDRet::fail;
56183ff9ab6SJames Feist                     }
56283ff9ab6SJames Feist                 }
56383ff9ab6SJames Feist 
56483ff9ab6SJames Feist                 // /refish/v1/chassis/chassis_name/
56583ff9ab6SJames Feist                 if (!dbus::utility::getNthStringFromPath(*chassisId, 3,
56683ff9ab6SJames Feist                                                          chassis))
56783ff9ab6SJames Feist                 {
56883ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Got invalid path " << *chassisId;
56935a62c7cSJason M. Bills                     messages::invalidObject(response->res, *chassisId);
57083ff9ab6SJames Feist                     return CreatePIDRet::fail;
57183ff9ab6SJames Feist                 }
57283ff9ab6SJames Feist             }
57383ff9ab6SJames Feist             else if (field.key() == "FailSafePercent" ||
57483ff9ab6SJames Feist                      field.key() == "MinThermalRpm")
57583ff9ab6SJames Feist             {
57683ff9ab6SJames Feist                 const double* ptr = field.value().get_ptr<const double*>();
57783ff9ab6SJames Feist                 if (ptr == nullptr)
57883ff9ab6SJames Feist                 {
57983ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
58035a62c7cSJason M. Bills                     messages::propertyValueFormatError(
58135a62c7cSJason M. Bills                         response->res, field.value().dump(), field.key());
58283ff9ab6SJames Feist                     return CreatePIDRet::fail;
58383ff9ab6SJames Feist                 }
58483ff9ab6SJames Feist                 output[field.key()] = *ptr;
58583ff9ab6SJames Feist             }
58683ff9ab6SJames Feist             else
58783ff9ab6SJames Feist             {
58883ff9ab6SJames Feist                 BMCWEB_LOG_ERROR << "Illegal Type " << field.key();
58935a62c7cSJason M. Bills                 messages::propertyUnknown(response->res, field.key());
59083ff9ab6SJames Feist                 return CreatePIDRet::fail;
59183ff9ab6SJames Feist             }
59283ff9ab6SJames Feist         }
59383ff9ab6SJames Feist     }
59483ff9ab6SJames Feist     else
59583ff9ab6SJames Feist     {
59683ff9ab6SJames Feist         BMCWEB_LOG_ERROR << "Illegal Type " << type;
59735a62c7cSJason M. Bills         messages::propertyUnknown(response->res, type);
59883ff9ab6SJames Feist         return CreatePIDRet::fail;
59983ff9ab6SJames Feist     }
60083ff9ab6SJames Feist     return CreatePIDRet::patch;
60183ff9ab6SJames Feist }
60283ff9ab6SJames Feist 
6031abe55efSEd Tanous class Manager : public Node
6041abe55efSEd Tanous {
6059c310685SBorawski.Lukasz   public:
6065b4aa86bSJames Feist     Manager(CrowApp& app) : Node(app, "/redfish/v1/Managers/bmc/")
6071abe55efSEd Tanous     {
6080f74e643SEd Tanous         uuid = app.template getMiddleware<crow::persistent_data::Middleware>()
60955c7b7a2SEd Tanous                    .systemUuid;
610a434f2bdSEd Tanous         entityPrivileges = {
611a434f2bdSEd Tanous             {boost::beast::http::verb::get, {{"Login"}}},
612e0d918bcSEd Tanous             {boost::beast::http::verb::head, {{"Login"}}},
613e0d918bcSEd Tanous             {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
614e0d918bcSEd Tanous             {boost::beast::http::verb::put, {{"ConfigureManager"}}},
615e0d918bcSEd Tanous             {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
616e0d918bcSEd Tanous             {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
6179c310685SBorawski.Lukasz     }
6189c310685SBorawski.Lukasz 
6199c310685SBorawski.Lukasz   private:
6205b4aa86bSJames Feist     void getPidValues(std::shared_ptr<AsyncResp> asyncResp)
6215b4aa86bSJames Feist     {
6225b4aa86bSJames Feist         crow::connections::systemBus->async_method_call(
6235b4aa86bSJames Feist             [asyncResp](const boost::system::error_code ec,
6245b4aa86bSJames Feist                         const crow::openbmc_mapper::GetSubTreeType& subtree) {
6255b4aa86bSJames Feist                 if (ec)
6265b4aa86bSJames Feist                 {
6275b4aa86bSJames Feist                     BMCWEB_LOG_ERROR << ec;
628f12894f8SJason M. Bills                     messages::internalError(asyncResp->res);
6295b4aa86bSJames Feist                     return;
6305b4aa86bSJames Feist                 }
6315b4aa86bSJames Feist 
6325b4aa86bSJames Feist                 // create map of <connection, path to objMgr>>
6335b4aa86bSJames Feist                 boost::container::flat_map<std::string, std::string>
6345b4aa86bSJames Feist                     objectMgrPaths;
6356bce33bcSJames Feist                 boost::container::flat_set<std::string> calledConnections;
6365b4aa86bSJames Feist                 for (const auto& pathGroup : subtree)
6375b4aa86bSJames Feist                 {
6385b4aa86bSJames Feist                     for (const auto& connectionGroup : pathGroup.second)
6395b4aa86bSJames Feist                     {
6406bce33bcSJames Feist                         auto findConnection =
6416bce33bcSJames Feist                             calledConnections.find(connectionGroup.first);
6426bce33bcSJames Feist                         if (findConnection != calledConnections.end())
6436bce33bcSJames Feist                         {
6446bce33bcSJames Feist                             break;
6456bce33bcSJames Feist                         }
6465b4aa86bSJames Feist                         for (const std::string& interface :
6475b4aa86bSJames Feist                              connectionGroup.second)
6485b4aa86bSJames Feist                         {
6495b4aa86bSJames Feist                             if (interface == objectManagerIface)
6505b4aa86bSJames Feist                             {
6515b4aa86bSJames Feist                                 objectMgrPaths[connectionGroup.first] =
6525b4aa86bSJames Feist                                     pathGroup.first;
6535b4aa86bSJames Feist                             }
6545b4aa86bSJames Feist                             // this list is alphabetical, so we
6555b4aa86bSJames Feist                             // should have found the objMgr by now
6565b4aa86bSJames Feist                             if (interface == pidConfigurationIface ||
6575b4aa86bSJames Feist                                 interface == pidZoneConfigurationIface)
6585b4aa86bSJames Feist                             {
6595b4aa86bSJames Feist                                 auto findObjMgr =
6605b4aa86bSJames Feist                                     objectMgrPaths.find(connectionGroup.first);
6615b4aa86bSJames Feist                                 if (findObjMgr == objectMgrPaths.end())
6625b4aa86bSJames Feist                                 {
6635b4aa86bSJames Feist                                     BMCWEB_LOG_DEBUG << connectionGroup.first
6645b4aa86bSJames Feist                                                      << "Has no Object Manager";
6655b4aa86bSJames Feist                                     continue;
6665b4aa86bSJames Feist                                 }
6676bce33bcSJames Feist 
6686bce33bcSJames Feist                                 calledConnections.insert(connectionGroup.first);
6696bce33bcSJames Feist 
6705b4aa86bSJames Feist                                 asyncPopulatePid(findObjMgr->first,
6715b4aa86bSJames Feist                                                  findObjMgr->second, asyncResp);
6725b4aa86bSJames Feist                                 break;
6735b4aa86bSJames Feist                             }
6745b4aa86bSJames Feist                         }
6755b4aa86bSJames Feist                     }
6765b4aa86bSJames Feist                 }
6775b4aa86bSJames Feist             },
6785b4aa86bSJames Feist             "xyz.openbmc_project.ObjectMapper",
6795b4aa86bSJames Feist             "/xyz/openbmc_project/object_mapper",
6805b4aa86bSJames Feist             "xyz.openbmc_project.ObjectMapper", "GetSubTree", "/", 0,
6815b4aa86bSJames Feist             std::array<const char*, 3>{pidConfigurationIface,
6825b4aa86bSJames Feist                                        pidZoneConfigurationIface,
6835b4aa86bSJames Feist                                        objectManagerIface});
6845b4aa86bSJames Feist     }
6855b4aa86bSJames Feist 
68655c7b7a2SEd Tanous     void doGet(crow::Response& res, const crow::Request& req,
6871abe55efSEd Tanous                const std::vector<std::string>& params) override
6881abe55efSEd Tanous     {
6890f74e643SEd Tanous         res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc";
6900f74e643SEd Tanous         res.jsonValue["@odata.type"] = "#Manager.v1_3_0.Manager";
6910f74e643SEd Tanous         res.jsonValue["@odata.context"] =
6920f74e643SEd Tanous             "/redfish/v1/$metadata#Manager.Manager";
6930f74e643SEd Tanous         res.jsonValue["Id"] = "bmc";
6940f74e643SEd Tanous         res.jsonValue["Name"] = "OpenBmc Manager";
6950f74e643SEd Tanous         res.jsonValue["Description"] = "Baseboard Management Controller";
6960f74e643SEd Tanous         res.jsonValue["PowerState"] = "On";
6970f74e643SEd Tanous         res.jsonValue["ManagerType"] = "BMC";
698*7517658eSEd Tanous         res.jsonValue["UUID"] = uuid;
6990f74e643SEd Tanous         res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model
7000f74e643SEd Tanous 
7010f74e643SEd Tanous         res.jsonValue["LogServices"] = {
7020f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/LogServices"}};
7030f74e643SEd Tanous 
7040f74e643SEd Tanous         res.jsonValue["NetworkProtocol"] = {
7050f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/NetworkProtocol"}};
7060f74e643SEd Tanous 
7070f74e643SEd Tanous         res.jsonValue["EthernetInterfaces"] = {
7080f74e643SEd Tanous             {"@odata.id", "/redfish/v1/Managers/bmc/EthernetInterfaces"}};
7090f74e643SEd Tanous         // default oem data
7100f74e643SEd Tanous         nlohmann::json& oem = res.jsonValue["Oem"];
7110f74e643SEd Tanous         nlohmann::json& oemOpenbmc = oem["OpenBmc"];
7120f74e643SEd Tanous         oem["@odata.type"] = "#OemManager.Oem";
7130f74e643SEd Tanous         oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem";
7140f74e643SEd Tanous         oem["@odata.context"] = "/redfish/v1/$metadata#OemManager.Oem";
7150f74e643SEd Tanous         oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc";
7160f74e643SEd Tanous         oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc";
7170f74e643SEd Tanous         oemOpenbmc["@odata.context"] =
7180f74e643SEd Tanous             "/redfish/v1/$metadata#OemManager.OpenBmc";
7190f74e643SEd Tanous 
720ed5befbdSJennifer Lee         // Update Actions object.
7210f74e643SEd Tanous         nlohmann::json& manager_reset =
7220f74e643SEd Tanous             res.jsonValue["Actions"]["#Manager.Reset"];
723ed5befbdSJennifer Lee         manager_reset["target"] =
724ed5befbdSJennifer Lee             "/redfish/v1/Managers/bmc/Actions/Manager.Reset";
725ed5befbdSJennifer Lee         manager_reset["ResetType@Redfish.AllowableValues"] = {
726ed5befbdSJennifer Lee             "GracefulRestart"};
727ca537928SJennifer Lee 
7280f74e643SEd Tanous         res.jsonValue["DateTime"] = getDateTime();
729ed5befbdSJennifer Lee         std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res);
7305b4aa86bSJames Feist 
731ca537928SJennifer Lee         crow::connections::systemBus->async_method_call(
732ca537928SJennifer Lee             [asyncResp](const boost::system::error_code ec,
7335b4aa86bSJames Feist                         const dbus::utility::ManagedObjectType& resp) {
734ca537928SJennifer Lee                 if (ec)
735ca537928SJennifer Lee                 {
736ca537928SJennifer Lee                     BMCWEB_LOG_ERROR << "Error while getting Software Version";
737f12894f8SJason M. Bills                     messages::internalError(asyncResp->res);
738ca537928SJennifer Lee                     return;
739ca537928SJennifer Lee                 }
740ca537928SJennifer Lee 
741ca537928SJennifer Lee                 for (auto& objpath : resp)
742ca537928SJennifer Lee                 {
743ca537928SJennifer Lee                     for (auto& interface : objpath.second)
744ca537928SJennifer Lee                     {
745ca537928SJennifer Lee                         // If interface is xyz.openbmc_project.Software.Version,
746ca537928SJennifer Lee                         // this is what we're looking for.
747ca537928SJennifer Lee                         if (interface.first ==
748ca537928SJennifer Lee                             "xyz.openbmc_project.Software.Version")
749ca537928SJennifer Lee                         {
750ca537928SJennifer Lee                             // Cut out everyting until last "/", ...
751ca537928SJennifer Lee                             const std::string& iface_id = objpath.first;
752ca537928SJennifer Lee                             for (auto& property : interface.second)
753ca537928SJennifer Lee                             {
754ca537928SJennifer Lee                                 if (property.first == "Version")
755ca537928SJennifer Lee                                 {
756ca537928SJennifer Lee                                     const std::string* value =
7571b6b96c5SEd Tanous                                         sdbusplus::message::variant_ns::get_if<
7581b6b96c5SEd Tanous                                             std::string>(&property.second);
759ca537928SJennifer Lee                                     if (value == nullptr)
760ca537928SJennifer Lee                                     {
761ca537928SJennifer Lee                                         continue;
762ca537928SJennifer Lee                                     }
763ca537928SJennifer Lee                                     asyncResp->res
764ca537928SJennifer Lee                                         .jsonValue["FirmwareVersion"] = *value;
765ca537928SJennifer Lee                                 }
766ca537928SJennifer Lee                             }
767ca537928SJennifer Lee                         }
768ca537928SJennifer Lee                     }
769ca537928SJennifer Lee                 }
770ca537928SJennifer Lee             },
771ca537928SJennifer Lee             "xyz.openbmc_project.Software.BMC.Updater",
772ca537928SJennifer Lee             "/xyz/openbmc_project/software",
773ca537928SJennifer Lee             "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
7745b4aa86bSJames Feist         getPidValues(asyncResp);
7755b4aa86bSJames Feist     }
77683ff9ab6SJames Feist     void setPidValues(std::shared_ptr<AsyncResp> response,
77783ff9ab6SJames Feist                       const nlohmann::json& data)
77883ff9ab6SJames Feist     {
77983ff9ab6SJames Feist         // todo(james): might make sense to do a mapper call here if this
78083ff9ab6SJames Feist         // interface gets more traction
78183ff9ab6SJames Feist         crow::connections::systemBus->async_method_call(
78283ff9ab6SJames Feist             [response,
78383ff9ab6SJames Feist              data](const boost::system::error_code ec,
78483ff9ab6SJames Feist                    const dbus::utility::ManagedObjectType& managedObj) {
78583ff9ab6SJames Feist                 if (ec)
78683ff9ab6SJames Feist                 {
78783ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Error communicating to Entity Manager";
78835a62c7cSJason M. Bills                     messages::internalError(response->res);
78983ff9ab6SJames Feist                     return;
79083ff9ab6SJames Feist                 }
79183ff9ab6SJames Feist                 for (const auto& type : data.items())
79283ff9ab6SJames Feist                 {
79383ff9ab6SJames Feist                     if (!type.value().is_object())
79483ff9ab6SJames Feist                     {
79583ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Illegal Type " << type.key();
79635a62c7cSJason M. Bills                         messages::propertyValueFormatError(
79735a62c7cSJason M. Bills                             response->res, type.value(), type.key());
79883ff9ab6SJames Feist                         return;
79983ff9ab6SJames Feist                     }
80083ff9ab6SJames Feist                     for (const auto& record : type.value().items())
80183ff9ab6SJames Feist                     {
80283ff9ab6SJames Feist                         const std::string& name = record.key();
80383ff9ab6SJames Feist                         auto pathItr =
80483ff9ab6SJames Feist                             std::find_if(managedObj.begin(), managedObj.end(),
80583ff9ab6SJames Feist                                          [&name](const auto& obj) {
80683ff9ab6SJames Feist                                              return boost::algorithm::ends_with(
80783ff9ab6SJames Feist                                                  obj.first.str, name);
80883ff9ab6SJames Feist                                          });
80983ff9ab6SJames Feist                         boost::container::flat_map<
81083ff9ab6SJames Feist                             std::string, dbus::utility::DbusVariantType>
81183ff9ab6SJames Feist                             output;
81283ff9ab6SJames Feist 
81383ff9ab6SJames Feist                         output.reserve(16); // The pid interface length
81483ff9ab6SJames Feist 
81583ff9ab6SJames Feist                         // determines if we're patching entity-manager or
81683ff9ab6SJames Feist                         // creating a new object
81783ff9ab6SJames Feist                         bool createNewObject = (pathItr == managedObj.end());
81883ff9ab6SJames Feist                         if (type.key() == "PidControllers" ||
81983ff9ab6SJames Feist                             type.key() == "FanControllers")
82083ff9ab6SJames Feist                         {
82183ff9ab6SJames Feist                             if (!createNewObject &&
82283ff9ab6SJames Feist                                 pathItr->second.find(pidConfigurationIface) ==
82383ff9ab6SJames Feist                                     pathItr->second.end())
82483ff9ab6SJames Feist                             {
82583ff9ab6SJames Feist                                 createNewObject = true;
82683ff9ab6SJames Feist                             }
82783ff9ab6SJames Feist                         }
82883ff9ab6SJames Feist                         else if (!createNewObject &&
82983ff9ab6SJames Feist                                  pathItr->second.find(
83083ff9ab6SJames Feist                                      pidZoneConfigurationIface) ==
83183ff9ab6SJames Feist                                      pathItr->second.end())
83283ff9ab6SJames Feist                         {
83383ff9ab6SJames Feist                             createNewObject = true;
83483ff9ab6SJames Feist                         }
83583ff9ab6SJames Feist                         output["Name"] =
83683ff9ab6SJames Feist                             boost::replace_all_copy(name, "_", " ");
83783ff9ab6SJames Feist 
83883ff9ab6SJames Feist                         std::string chassis;
83983ff9ab6SJames Feist                         CreatePIDRet ret = createPidInterface(
84083ff9ab6SJames Feist                             response, type.key(), record.value(),
84183ff9ab6SJames Feist                             pathItr->first.str, managedObj, createNewObject,
84283ff9ab6SJames Feist                             output, chassis);
84383ff9ab6SJames Feist                         if (ret == CreatePIDRet::fail)
84483ff9ab6SJames Feist                         {
84583ff9ab6SJames Feist                             return;
84683ff9ab6SJames Feist                         }
84783ff9ab6SJames Feist                         else if (ret == CreatePIDRet::del)
84883ff9ab6SJames Feist                         {
84983ff9ab6SJames Feist                             continue;
85083ff9ab6SJames Feist                         }
85183ff9ab6SJames Feist 
85283ff9ab6SJames Feist                         if (!createNewObject)
85383ff9ab6SJames Feist                         {
85483ff9ab6SJames Feist                             for (const auto& property : output)
85583ff9ab6SJames Feist                             {
85683ff9ab6SJames Feist                                 const char* iface =
85783ff9ab6SJames Feist                                     type.key() == "FanZones"
85883ff9ab6SJames Feist                                         ? pidZoneConfigurationIface
85983ff9ab6SJames Feist                                         : pidConfigurationIface;
86083ff9ab6SJames Feist                                 crow::connections::systemBus->async_method_call(
86183ff9ab6SJames Feist                                     [response,
86283ff9ab6SJames Feist                                      propertyName{std::string(property.first)}](
86383ff9ab6SJames Feist                                         const boost::system::error_code ec) {
86483ff9ab6SJames Feist                                         if (ec)
86583ff9ab6SJames Feist                                         {
86683ff9ab6SJames Feist                                             BMCWEB_LOG_ERROR
86783ff9ab6SJames Feist                                                 << "Error patching "
86883ff9ab6SJames Feist                                                 << propertyName << ": " << ec;
86935a62c7cSJason M. Bills                                             messages::internalError(
87035a62c7cSJason M. Bills                                                 response->res);
87183ff9ab6SJames Feist                                         }
87283ff9ab6SJames Feist                                     },
87383ff9ab6SJames Feist                                     "xyz.openbmc_project.EntityManager",
87483ff9ab6SJames Feist                                     pathItr->first.str,
87583ff9ab6SJames Feist                                     "org.freedesktop.DBus.Properties", "Set",
87683ff9ab6SJames Feist                                     std::string(iface), property.first,
87783ff9ab6SJames Feist                                     property.second);
87883ff9ab6SJames Feist                             }
87983ff9ab6SJames Feist                         }
88083ff9ab6SJames Feist                         else
88183ff9ab6SJames Feist                         {
88283ff9ab6SJames Feist                             if (chassis.empty())
88383ff9ab6SJames Feist                             {
88483ff9ab6SJames Feist                                 BMCWEB_LOG_ERROR
88583ff9ab6SJames Feist                                     << "Failed to get chassis from config";
88635a62c7cSJason M. Bills                                 messages::invalidObject(response->res, name);
88783ff9ab6SJames Feist                                 return;
88883ff9ab6SJames Feist                             }
88983ff9ab6SJames Feist 
89083ff9ab6SJames Feist                             bool foundChassis = false;
89183ff9ab6SJames Feist                             for (const auto& obj : managedObj)
89283ff9ab6SJames Feist                             {
89383ff9ab6SJames Feist                                 if (boost::algorithm::ends_with(obj.first.str,
89483ff9ab6SJames Feist                                                                 chassis))
89583ff9ab6SJames Feist                                 {
89683ff9ab6SJames Feist                                     chassis = obj.first.str;
89783ff9ab6SJames Feist                                     foundChassis = true;
89883ff9ab6SJames Feist                                     break;
89983ff9ab6SJames Feist                                 }
90083ff9ab6SJames Feist                             }
90183ff9ab6SJames Feist                             if (!foundChassis)
90283ff9ab6SJames Feist                             {
90383ff9ab6SJames Feist                                 BMCWEB_LOG_ERROR
90483ff9ab6SJames Feist                                     << "Failed to find chassis on dbus";
90583ff9ab6SJames Feist                                 messages::resourceMissingAtURI(
90635a62c7cSJason M. Bills                                     response->res,
90735a62c7cSJason M. Bills                                     "/redfish/v1/Chassis/" + chassis);
90883ff9ab6SJames Feist                                 return;
90983ff9ab6SJames Feist                             }
91083ff9ab6SJames Feist 
91183ff9ab6SJames Feist                             crow::connections::systemBus->async_method_call(
91283ff9ab6SJames Feist                                 [response](const boost::system::error_code ec) {
91383ff9ab6SJames Feist                                     if (ec)
91483ff9ab6SJames Feist                                     {
91583ff9ab6SJames Feist                                         BMCWEB_LOG_ERROR
91683ff9ab6SJames Feist                                             << "Error Adding Pid Object " << ec;
91735a62c7cSJason M. Bills                                         messages::internalError(response->res);
91883ff9ab6SJames Feist                                     }
91983ff9ab6SJames Feist                                 },
92083ff9ab6SJames Feist                                 "xyz.openbmc_project.EntityManager", chassis,
92183ff9ab6SJames Feist                                 "xyz.openbmc_project.AddObject", "AddObject",
92283ff9ab6SJames Feist                                 output);
92383ff9ab6SJames Feist                         }
92483ff9ab6SJames Feist                     }
92583ff9ab6SJames Feist                 }
92683ff9ab6SJames Feist             },
92783ff9ab6SJames Feist             "xyz.openbmc_project.EntityManager", "/", objectManagerIface,
92883ff9ab6SJames Feist             "GetManagedObjects");
92983ff9ab6SJames Feist     }
9305b4aa86bSJames Feist 
9315b4aa86bSJames Feist     void doPatch(crow::Response& res, const crow::Request& req,
9325b4aa86bSJames Feist                  const std::vector<std::string>& params) override
9335b4aa86bSJames Feist     {
9340627a2c7SEd Tanous         std::optional<nlohmann::json> oem;
9350627a2c7SEd Tanous 
9360627a2c7SEd Tanous         if (!json_util::readJson(req, res, "Oem", oem))
93783ff9ab6SJames Feist         {
93883ff9ab6SJames Feist             return;
93983ff9ab6SJames Feist         }
9400627a2c7SEd Tanous 
94183ff9ab6SJames Feist         std::shared_ptr<AsyncResp> response = std::make_shared<AsyncResp>(res);
9420627a2c7SEd Tanous 
9430627a2c7SEd Tanous         if (oem)
94483ff9ab6SJames Feist         {
9450627a2c7SEd Tanous             for (const auto& oemLevel : oem->items())
94683ff9ab6SJames Feist             {
94783ff9ab6SJames Feist                 if (oemLevel.key() == "OpenBmc")
94883ff9ab6SJames Feist                 {
94983ff9ab6SJames Feist                     if (!oemLevel.value().is_object())
95083ff9ab6SJames Feist                     {
95183ff9ab6SJames Feist                         BMCWEB_LOG_ERROR << "Bad Patch " << oemLevel.key();
95235a62c7cSJason M. Bills                         messages::propertyValueFormatError(
9530627a2c7SEd Tanous                             response->res, "Oem", "OemManager.OpenBmc");
95483ff9ab6SJames Feist                         return;
95583ff9ab6SJames Feist                     }
95683ff9ab6SJames Feist                     for (const auto& typeLevel : oemLevel.value().items())
95783ff9ab6SJames Feist                     {
95883ff9ab6SJames Feist 
95983ff9ab6SJames Feist                         if (typeLevel.key() == "Fan")
96083ff9ab6SJames Feist                         {
96183ff9ab6SJames Feist                             if (!typeLevel.value().is_object())
96283ff9ab6SJames Feist                             {
96383ff9ab6SJames Feist                                 BMCWEB_LOG_ERROR << "Bad Patch "
96483ff9ab6SJames Feist                                                  << typeLevel.key();
96583ff9ab6SJames Feist                                 messages::propertyValueFormatError(
96635a62c7cSJason M. Bills                                     response->res, typeLevel.value().dump(),
96735a62c7cSJason M. Bills                                     typeLevel.key());
96883ff9ab6SJames Feist                                 return;
96983ff9ab6SJames Feist                             }
97083ff9ab6SJames Feist                             setPidValues(response,
97183ff9ab6SJames Feist                                          std::move(typeLevel.value()));
97283ff9ab6SJames Feist                         }
97383ff9ab6SJames Feist                         else
97483ff9ab6SJames Feist                         {
97583ff9ab6SJames Feist                             BMCWEB_LOG_ERROR << "Bad Patch " << typeLevel.key();
97635a62c7cSJason M. Bills                             messages::propertyUnknown(response->res,
97735a62c7cSJason M. Bills                                                       typeLevel.key());
97883ff9ab6SJames Feist                             return;
97983ff9ab6SJames Feist                         }
98083ff9ab6SJames Feist                     }
98183ff9ab6SJames Feist                 }
98283ff9ab6SJames Feist                 else
98383ff9ab6SJames Feist                 {
98483ff9ab6SJames Feist                     BMCWEB_LOG_ERROR << "Bad Patch " << oemLevel.key();
98535a62c7cSJason M. Bills                     messages::propertyUnknown(response->res, oemLevel.key());
98683ff9ab6SJames Feist                     return;
98783ff9ab6SJames Feist                 }
98883ff9ab6SJames Feist             }
98983ff9ab6SJames Feist         }
9909c310685SBorawski.Lukasz     }
9919c310685SBorawski.Lukasz 
9921abe55efSEd Tanous     std::string getDateTime() const
9931abe55efSEd Tanous     {
9949c310685SBorawski.Lukasz         std::array<char, 128> dateTime;
9959c310685SBorawski.Lukasz         std::string redfishDateTime("0000-00-00T00:00:00Z00:00");
9969c310685SBorawski.Lukasz         std::time_t time = std::time(nullptr);
9979c310685SBorawski.Lukasz 
9989c310685SBorawski.Lukasz         if (std::strftime(dateTime.begin(), dateTime.size(), "%FT%T%z",
9991abe55efSEd Tanous                           std::localtime(&time)))
10001abe55efSEd Tanous         {
10019c310685SBorawski.Lukasz             // insert the colon required by the ISO 8601 standard
10029c310685SBorawski.Lukasz             redfishDateTime = std::string(dateTime.data());
10039c310685SBorawski.Lukasz             redfishDateTime.insert(redfishDateTime.end() - 2, ':');
10049c310685SBorawski.Lukasz         }
10059c310685SBorawski.Lukasz 
10069c310685SBorawski.Lukasz         return redfishDateTime;
10079c310685SBorawski.Lukasz     }
10080f74e643SEd Tanous 
10090f74e643SEd Tanous     std::string uuid;
10109c310685SBorawski.Lukasz };
10119c310685SBorawski.Lukasz 
10121abe55efSEd Tanous class ManagerCollection : public Node
10131abe55efSEd Tanous {
10149c310685SBorawski.Lukasz   public:
10151abe55efSEd Tanous     ManagerCollection(CrowApp& app) : Node(app, "/redfish/v1/Managers/")
10161abe55efSEd Tanous     {
1017a434f2bdSEd Tanous         entityPrivileges = {
1018a434f2bdSEd Tanous             {boost::beast::http::verb::get, {{"Login"}}},
1019e0d918bcSEd Tanous             {boost::beast::http::verb::head, {{"Login"}}},
1020e0d918bcSEd Tanous             {boost::beast::http::verb::patch, {{"ConfigureManager"}}},
1021e0d918bcSEd Tanous             {boost::beast::http::verb::put, {{"ConfigureManager"}}},
1022e0d918bcSEd Tanous             {boost::beast::http::verb::delete_, {{"ConfigureManager"}}},
1023e0d918bcSEd Tanous             {boost::beast::http::verb::post, {{"ConfigureManager"}}}};
10249c310685SBorawski.Lukasz     }
10259c310685SBorawski.Lukasz 
10269c310685SBorawski.Lukasz   private:
102755c7b7a2SEd Tanous     void doGet(crow::Response& res, const crow::Request& req,
10281abe55efSEd Tanous                const std::vector<std::string>& params) override
10291abe55efSEd Tanous     {
103083ff9ab6SJames Feist         // Collections don't include the static data added by SubRoute
103183ff9ab6SJames Feist         // because it has a duplicate entry for members
103255c7b7a2SEd Tanous         res.jsonValue["@odata.id"] = "/redfish/v1/Managers";
103355c7b7a2SEd Tanous         res.jsonValue["@odata.type"] = "#ManagerCollection.ManagerCollection";
103455c7b7a2SEd Tanous         res.jsonValue["@odata.context"] =
103555c7b7a2SEd Tanous             "/redfish/v1/$metadata#ManagerCollection.ManagerCollection";
103655c7b7a2SEd Tanous         res.jsonValue["Name"] = "Manager Collection";
103755c7b7a2SEd Tanous         res.jsonValue["Members@odata.count"] = 1;
103855c7b7a2SEd Tanous         res.jsonValue["Members"] = {
10395b4aa86bSJames Feist             {{"@odata.id", "/redfish/v1/Managers/bmc"}}};
10409c310685SBorawski.Lukasz         res.end();
10419c310685SBorawski.Lukasz     }
10429c310685SBorawski.Lukasz };
10439c310685SBorawski.Lukasz } // namespace redfish
1044