xref: /openbmc/bmcweb/features/redfish/lib/managers.hpp (revision cddbf3df0ea0b78521b62f30f2ce764c6619018a)
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 
18a51fc2d2SSui Chen #include "app.hpp"
19a51fc2d2SSui Chen #include "dbus_utility.hpp"
20b49ac873SJames Feist #include "health.hpp"
21a51fc2d2SSui Chen #include "query.hpp"
22c5d03ff4SJennifer Lee #include "redfish_util.hpp"
23a51fc2d2SSui Chen #include "registries/privilege_registry.hpp"
24fac6e53bSKrzysztof Grobelny #include "utils/dbus_utils.hpp"
253ccb3adbSEd Tanous #include "utils/json_utils.hpp"
26a51fc2d2SSui Chen #include "utils/sw_utils.hpp"
27a51fc2d2SSui Chen #include "utils/systemd_utils.hpp"
282b82937eSEd Tanous #include "utils/time_utils.hpp"
299c310685SBorawski.Lukasz 
30e99073f5SGeorge Liu #include <boost/system/error_code.hpp>
31fac6e53bSKrzysztof Grobelny #include <sdbusplus/asio/property.hpp>
32fac6e53bSKrzysztof Grobelny #include <sdbusplus/unpack_properties.hpp>
331214b7e7SGunnar Mills 
34a170f275SEd Tanous #include <algorithm>
35e99073f5SGeorge Liu #include <array>
364bfefa74SGunnar Mills #include <cstdint>
371214b7e7SGunnar Mills #include <memory>
381214b7e7SGunnar Mills #include <sstream>
39e99073f5SGeorge Liu #include <string_view>
40abf2add6SEd Tanous #include <variant>
415b4aa86bSJames Feist 
421abe55efSEd Tanous namespace redfish
431abe55efSEd Tanous {
44ed5befbdSJennifer Lee 
45ed5befbdSJennifer Lee /**
462a5c4407SGunnar Mills  * Function reboots the BMC.
472a5c4407SGunnar Mills  *
482a5c4407SGunnar Mills  * @param[in] asyncResp - Shared pointer for completing asynchronous calls
49ed5befbdSJennifer Lee  */
508d1b46d7Szhanghch05 inline void
518d1b46d7Szhanghch05     doBMCGracefulRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
52ed5befbdSJennifer Lee {
53ed5befbdSJennifer Lee     const char* processName = "xyz.openbmc_project.State.BMC";
54ed5befbdSJennifer Lee     const char* objectPath = "/xyz/openbmc_project/state/bmc0";
55ed5befbdSJennifer Lee     const char* interfaceName = "xyz.openbmc_project.State.BMC";
56ed5befbdSJennifer Lee     const std::string& propertyValue =
57ed5befbdSJennifer Lee         "xyz.openbmc_project.State.BMC.Transition.Reboot";
58ed5befbdSJennifer Lee     const char* destProperty = "RequestedBMCTransition";
59ed5befbdSJennifer Lee 
60ed5befbdSJennifer Lee     // Create the D-Bus variant for D-Bus call.
61168e20c1SEd Tanous     dbus::utility::DbusVariantType dbusPropertyValue(propertyValue);
62ed5befbdSJennifer Lee 
63ed5befbdSJennifer Lee     crow::connections::systemBus->async_method_call(
645e7e2dc5SEd Tanous         [asyncResp](const boost::system::error_code& ec) {
65ed5befbdSJennifer Lee         // Use "Set" method to set the property value.
66ed5befbdSJennifer Lee         if (ec)
67ed5befbdSJennifer Lee         {
682a5c4407SGunnar Mills             BMCWEB_LOG_DEBUG << "[Set] Bad D-Bus request error: " << ec;
69ed5befbdSJennifer Lee             messages::internalError(asyncResp->res);
70ed5befbdSJennifer Lee             return;
71ed5befbdSJennifer Lee         }
72ed5befbdSJennifer Lee 
73ed5befbdSJennifer Lee         messages::success(asyncResp->res);
74ed5befbdSJennifer Lee         },
75ed5befbdSJennifer Lee         processName, objectPath, "org.freedesktop.DBus.Properties", "Set",
76ed5befbdSJennifer Lee         interfaceName, destProperty, dbusPropertyValue);
77ed5befbdSJennifer Lee }
782a5c4407SGunnar Mills 
798d1b46d7Szhanghch05 inline void
808d1b46d7Szhanghch05     doBMCForceRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
81f92af389SJayaprakash Mutyala {
82f92af389SJayaprakash Mutyala     const char* processName = "xyz.openbmc_project.State.BMC";
83f92af389SJayaprakash Mutyala     const char* objectPath = "/xyz/openbmc_project/state/bmc0";
84f92af389SJayaprakash Mutyala     const char* interfaceName = "xyz.openbmc_project.State.BMC";
85f92af389SJayaprakash Mutyala     const std::string& propertyValue =
86f92af389SJayaprakash Mutyala         "xyz.openbmc_project.State.BMC.Transition.HardReboot";
87f92af389SJayaprakash Mutyala     const char* destProperty = "RequestedBMCTransition";
88f92af389SJayaprakash Mutyala 
89f92af389SJayaprakash Mutyala     // Create the D-Bus variant for D-Bus call.
90168e20c1SEd Tanous     dbus::utility::DbusVariantType dbusPropertyValue(propertyValue);
91f92af389SJayaprakash Mutyala 
92f92af389SJayaprakash Mutyala     crow::connections::systemBus->async_method_call(
935e7e2dc5SEd Tanous         [asyncResp](const boost::system::error_code& ec) {
94f92af389SJayaprakash Mutyala         // Use "Set" method to set the property value.
95f92af389SJayaprakash Mutyala         if (ec)
96f92af389SJayaprakash Mutyala         {
97f92af389SJayaprakash Mutyala             BMCWEB_LOG_DEBUG << "[Set] Bad D-Bus request error: " << ec;
98f92af389SJayaprakash Mutyala             messages::internalError(asyncResp->res);
99f92af389SJayaprakash Mutyala             return;
100f92af389SJayaprakash Mutyala         }
101f92af389SJayaprakash Mutyala 
102f92af389SJayaprakash Mutyala         messages::success(asyncResp->res);
103f92af389SJayaprakash Mutyala         },
104f92af389SJayaprakash Mutyala         processName, objectPath, "org.freedesktop.DBus.Properties", "Set",
105f92af389SJayaprakash Mutyala         interfaceName, destProperty, dbusPropertyValue);
106f92af389SJayaprakash Mutyala }
107f92af389SJayaprakash Mutyala 
1082a5c4407SGunnar Mills /**
1092a5c4407SGunnar Mills  * ManagerResetAction class supports the POST method for the Reset (reboot)
1102a5c4407SGunnar Mills  * action.
1112a5c4407SGunnar Mills  */
1127e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetAction(App& app)
1132a5c4407SGunnar Mills {
1142a5c4407SGunnar Mills     /**
1152a5c4407SGunnar Mills      * Function handles POST method request.
1162a5c4407SGunnar Mills      * Analyzes POST body before sending Reset (Reboot) request data to D-Bus.
117f92af389SJayaprakash Mutyala      * OpenBMC supports ResetType "GracefulRestart" and "ForceRestart".
1182a5c4407SGunnar Mills      */
1197e860f15SJohn Edward Broadbent 
1207e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/")
121ed398213SEd Tanous         .privileges(redfish::privileges::postManager)
1227e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::post)(
12345ca1b86SEd Tanous             [&app](const crow::Request& req,
1247e860f15SJohn Edward Broadbent                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
1253ba00073SCarson Labrado         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
12645ca1b86SEd Tanous         {
12745ca1b86SEd Tanous             return;
12845ca1b86SEd Tanous         }
1292a5c4407SGunnar Mills         BMCWEB_LOG_DEBUG << "Post Manager Reset.";
1302a5c4407SGunnar Mills 
1312a5c4407SGunnar Mills         std::string resetType;
1322a5c4407SGunnar Mills 
13315ed6780SWilly Tu         if (!json_util::readJsonAction(req, asyncResp->res, "ResetType",
1347e860f15SJohn Edward Broadbent                                        resetType))
1352a5c4407SGunnar Mills         {
1362a5c4407SGunnar Mills             return;
1372a5c4407SGunnar Mills         }
1382a5c4407SGunnar Mills 
139f92af389SJayaprakash Mutyala         if (resetType == "GracefulRestart")
140f92af389SJayaprakash Mutyala         {
141f92af389SJayaprakash Mutyala             BMCWEB_LOG_DEBUG << "Proceeding with " << resetType;
142f92af389SJayaprakash Mutyala             doBMCGracefulRestart(asyncResp);
143f92af389SJayaprakash Mutyala             return;
144f92af389SJayaprakash Mutyala         }
1453174e4dfSEd Tanous         if (resetType == "ForceRestart")
146f92af389SJayaprakash Mutyala         {
147f92af389SJayaprakash Mutyala             BMCWEB_LOG_DEBUG << "Proceeding with " << resetType;
148f92af389SJayaprakash Mutyala             doBMCForceRestart(asyncResp);
149f92af389SJayaprakash Mutyala             return;
150f92af389SJayaprakash Mutyala         }
1512a5c4407SGunnar Mills         BMCWEB_LOG_DEBUG << "Invalid property value for ResetType: "
1522a5c4407SGunnar Mills                          << resetType;
1532a5c4407SGunnar Mills         messages::actionParameterNotSupported(asyncResp->res, resetType,
1542a5c4407SGunnar Mills                                               "ResetType");
1552a5c4407SGunnar Mills 
1562a5c4407SGunnar Mills         return;
1577e860f15SJohn Edward Broadbent         });
1582a5c4407SGunnar Mills }
159ed5befbdSJennifer Lee 
1603e40fc74SGunnar Mills /**
1613e40fc74SGunnar Mills  * ManagerResetToDefaultsAction class supports POST method for factory reset
1623e40fc74SGunnar Mills  * action.
1633e40fc74SGunnar Mills  */
1647e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetToDefaultsAction(App& app)
1653e40fc74SGunnar Mills {
1663e40fc74SGunnar Mills 
1673e40fc74SGunnar Mills     /**
1683e40fc74SGunnar Mills      * Function handles ResetToDefaults POST method request.
1693e40fc74SGunnar Mills      *
1703e40fc74SGunnar Mills      * Analyzes POST body message and factory resets BMC by calling
1713e40fc74SGunnar Mills      * BMC code updater factory reset followed by a BMC reboot.
1723e40fc74SGunnar Mills      *
1733e40fc74SGunnar Mills      * BMC code updater factory reset wipes the whole BMC read-write
1743e40fc74SGunnar Mills      * filesystem which includes things like the network settings.
1753e40fc74SGunnar Mills      *
1763e40fc74SGunnar Mills      * OpenBMC only supports ResetToDefaultsType "ResetAll".
1773e40fc74SGunnar Mills      */
1787e860f15SJohn Edward Broadbent 
1797e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app,
1807e860f15SJohn Edward Broadbent                  "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults/")
181ed398213SEd Tanous         .privileges(redfish::privileges::postManager)
1827e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::post)(
18345ca1b86SEd Tanous             [&app](const crow::Request& req,
1847e860f15SJohn Edward Broadbent                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
1853ba00073SCarson Labrado         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
18645ca1b86SEd Tanous         {
18745ca1b86SEd Tanous             return;
18845ca1b86SEd Tanous         }
1893e40fc74SGunnar Mills         BMCWEB_LOG_DEBUG << "Post ResetToDefaults.";
1903e40fc74SGunnar Mills 
1913e40fc74SGunnar Mills         std::string resetType;
1923e40fc74SGunnar Mills 
193002d39b4SEd Tanous         if (!json_util::readJsonAction(req, asyncResp->res,
194002d39b4SEd Tanous                                        "ResetToDefaultsType", resetType))
1953e40fc74SGunnar Mills         {
1963e40fc74SGunnar Mills             BMCWEB_LOG_DEBUG << "Missing property ResetToDefaultsType.";
1973e40fc74SGunnar Mills 
198002d39b4SEd Tanous             messages::actionParameterMissing(asyncResp->res, "ResetToDefaults",
1993e40fc74SGunnar Mills                                              "ResetToDefaultsType");
2003e40fc74SGunnar Mills             return;
2013e40fc74SGunnar Mills         }
2023e40fc74SGunnar Mills 
2033e40fc74SGunnar Mills         if (resetType != "ResetAll")
2043e40fc74SGunnar Mills         {
2050fda0f12SGeorge Liu             BMCWEB_LOG_DEBUG
2060fda0f12SGeorge Liu                 << "Invalid property value for ResetToDefaultsType: "
2073e40fc74SGunnar Mills                 << resetType;
208002d39b4SEd Tanous             messages::actionParameterNotSupported(asyncResp->res, resetType,
209002d39b4SEd Tanous                                                   "ResetToDefaultsType");
2103e40fc74SGunnar Mills             return;
2113e40fc74SGunnar Mills         }
2123e40fc74SGunnar Mills 
2133e40fc74SGunnar Mills         crow::connections::systemBus->async_method_call(
2145e7e2dc5SEd Tanous             [asyncResp](const boost::system::error_code& ec) {
2153e40fc74SGunnar Mills             if (ec)
2163e40fc74SGunnar Mills             {
217002d39b4SEd Tanous                 BMCWEB_LOG_DEBUG << "Failed to ResetToDefaults: " << ec;
2183e40fc74SGunnar Mills                 messages::internalError(asyncResp->res);
2193e40fc74SGunnar Mills                 return;
2203e40fc74SGunnar Mills             }
2213e40fc74SGunnar Mills             // Factory Reset doesn't actually happen until a reboot
2223e40fc74SGunnar Mills             // Can't erase what the BMC is running on
2233e40fc74SGunnar Mills             doBMCGracefulRestart(asyncResp);
2243e40fc74SGunnar Mills             },
2253e40fc74SGunnar Mills             "xyz.openbmc_project.Software.BMC.Updater",
2263e40fc74SGunnar Mills             "/xyz/openbmc_project/software",
2273e40fc74SGunnar Mills             "xyz.openbmc_project.Common.FactoryReset", "Reset");
2287e860f15SJohn Edward Broadbent         });
2293e40fc74SGunnar Mills }
2303e40fc74SGunnar Mills 
2311cb1a9e6SAppaRao Puli /**
2321cb1a9e6SAppaRao Puli  * ManagerResetActionInfo derived class for delivering Manager
2331cb1a9e6SAppaRao Puli  * ResetType AllowableValues using ResetInfo schema.
2341cb1a9e6SAppaRao Puli  */
2357e860f15SJohn Edward Broadbent inline void requestRoutesManagerResetActionInfo(App& app)
2361cb1a9e6SAppaRao Puli {
2371cb1a9e6SAppaRao Puli     /**
2381cb1a9e6SAppaRao Puli      * Functions triggers appropriate requests on DBus
2391cb1a9e6SAppaRao Puli      */
2407e860f15SJohn Edward Broadbent 
2417e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/ResetActionInfo/")
242ed398213SEd Tanous         .privileges(redfish::privileges::getActionInfo)
2437e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::get)(
24445ca1b86SEd Tanous             [&app](const crow::Request& req,
2457e860f15SJohn Edward Broadbent                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2463ba00073SCarson Labrado         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
24745ca1b86SEd Tanous         {
24845ca1b86SEd Tanous             return;
24945ca1b86SEd Tanous         }
2501476687dSEd Tanous 
2511476687dSEd Tanous         asyncResp->res.jsonValue["@odata.type"] =
2521476687dSEd Tanous             "#ActionInfo.v1_1_2.ActionInfo";
2531476687dSEd Tanous         asyncResp->res.jsonValue["@odata.id"] =
2541476687dSEd Tanous             "/redfish/v1/Managers/bmc/ResetActionInfo";
2551476687dSEd Tanous         asyncResp->res.jsonValue["Name"] = "Reset Action Info";
2561476687dSEd Tanous         asyncResp->res.jsonValue["Id"] = "ResetActionInfo";
2571476687dSEd Tanous         nlohmann::json::object_t parameter;
2581476687dSEd Tanous         parameter["Name"] = "ResetType";
2591476687dSEd Tanous         parameter["Required"] = true;
2601476687dSEd Tanous         parameter["DataType"] = "String";
2611476687dSEd Tanous 
2621476687dSEd Tanous         nlohmann::json::array_t allowableValues;
2631476687dSEd Tanous         allowableValues.push_back("GracefulRestart");
2641476687dSEd Tanous         allowableValues.push_back("ForceRestart");
2651476687dSEd Tanous         parameter["AllowableValues"] = std::move(allowableValues);
2661476687dSEd Tanous 
2671476687dSEd Tanous         nlohmann::json::array_t parameters;
2681476687dSEd Tanous         parameters.push_back(std::move(parameter));
2691476687dSEd Tanous 
2701476687dSEd Tanous         asyncResp->res.jsonValue["Parameters"] = std::move(parameters);
2717e860f15SJohn Edward Broadbent         });
2721cb1a9e6SAppaRao Puli }
2731cb1a9e6SAppaRao Puli 
2745b4aa86bSJames Feist static constexpr const char* objectManagerIface =
2755b4aa86bSJames Feist     "org.freedesktop.DBus.ObjectManager";
2765b4aa86bSJames Feist static constexpr const char* pidConfigurationIface =
2775b4aa86bSJames Feist     "xyz.openbmc_project.Configuration.Pid";
2785b4aa86bSJames Feist static constexpr const char* pidZoneConfigurationIface =
2795b4aa86bSJames Feist     "xyz.openbmc_project.Configuration.Pid.Zone";
280b7a08d04SJames Feist static constexpr const char* stepwiseConfigurationIface =
281b7a08d04SJames Feist     "xyz.openbmc_project.Configuration.Stepwise";
28273df0db0SJames Feist static constexpr const char* thermalModeIface =
28373df0db0SJames Feist     "xyz.openbmc_project.Control.ThermalMode";
2849c310685SBorawski.Lukasz 
2858d1b46d7Szhanghch05 inline void
2868d1b46d7Szhanghch05     asyncPopulatePid(const std::string& connection, const std::string& path,
28773df0db0SJames Feist                      const std::string& currentProfile,
28873df0db0SJames Feist                      const std::vector<std::string>& supportedProfiles,
2898d1b46d7Szhanghch05                      const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
2905b4aa86bSJames Feist {
2915b4aa86bSJames Feist 
2925b4aa86bSJames Feist     crow::connections::systemBus->async_method_call(
29373df0db0SJames Feist         [asyncResp, currentProfile, supportedProfiles](
2945e7e2dc5SEd Tanous             const boost::system::error_code& ec,
2955b4aa86bSJames Feist             const dbus::utility::ManagedObjectType& managedObj) {
2965b4aa86bSJames Feist         if (ec)
2975b4aa86bSJames Feist         {
2985b4aa86bSJames Feist             BMCWEB_LOG_ERROR << ec;
2995b4aa86bSJames Feist             asyncResp->res.jsonValue.clear();
300f12894f8SJason M. Bills             messages::internalError(asyncResp->res);
3015b4aa86bSJames Feist             return;
3025b4aa86bSJames Feist         }
3035b4aa86bSJames Feist         nlohmann::json& configRoot =
3045b4aa86bSJames Feist             asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"];
3055b4aa86bSJames Feist         nlohmann::json& fans = configRoot["FanControllers"];
3065b4aa86bSJames Feist         fans["@odata.type"] = "#OemManager.FanControllers";
3070fda0f12SGeorge Liu         fans["@odata.id"] =
3080fda0f12SGeorge Liu             "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanControllers";
3095b4aa86bSJames Feist 
3105b4aa86bSJames Feist         nlohmann::json& pids = configRoot["PidControllers"];
3115b4aa86bSJames Feist         pids["@odata.type"] = "#OemManager.PidControllers";
3125b4aa86bSJames Feist         pids["@odata.id"] =
3135b4aa86bSJames Feist             "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers";
3145b4aa86bSJames Feist 
315b7a08d04SJames Feist         nlohmann::json& stepwise = configRoot["StepwiseControllers"];
316b7a08d04SJames Feist         stepwise["@odata.type"] = "#OemManager.StepwiseControllers";
317b7a08d04SJames Feist         stepwise["@odata.id"] =
318b7a08d04SJames Feist             "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers";
319b7a08d04SJames Feist 
3205b4aa86bSJames Feist         nlohmann::json& zones = configRoot["FanZones"];
3215b4aa86bSJames Feist         zones["@odata.id"] =
3225b4aa86bSJames Feist             "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones";
3235b4aa86bSJames Feist         zones["@odata.type"] = "#OemManager.FanZones";
324002d39b4SEd Tanous         configRoot["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan";
3255b4aa86bSJames Feist         configRoot["@odata.type"] = "#OemManager.Fan";
32673df0db0SJames Feist         configRoot["Profile@Redfish.AllowableValues"] = supportedProfiles;
32773df0db0SJames Feist 
32873df0db0SJames Feist         if (!currentProfile.empty())
32973df0db0SJames Feist         {
33073df0db0SJames Feist             configRoot["Profile"] = currentProfile;
33173df0db0SJames Feist         }
33273df0db0SJames Feist         BMCWEB_LOG_ERROR << "profile = " << currentProfile << " !";
3335b4aa86bSJames Feist 
3345b4aa86bSJames Feist         for (const auto& pathPair : managedObj)
3355b4aa86bSJames Feist         {
3365b4aa86bSJames Feist             for (const auto& intfPair : pathPair.second)
3375b4aa86bSJames Feist             {
3385b4aa86bSJames Feist                 if (intfPair.first != pidConfigurationIface &&
339b7a08d04SJames Feist                     intfPair.first != pidZoneConfigurationIface &&
340b7a08d04SJames Feist                     intfPair.first != stepwiseConfigurationIface)
3415b4aa86bSJames Feist                 {
3425b4aa86bSJames Feist                     continue;
3435b4aa86bSJames Feist                 }
34473df0db0SJames Feist 
345711ac7a9SEd Tanous                 std::string name;
346711ac7a9SEd Tanous 
347711ac7a9SEd Tanous                 for (const std::pair<std::string,
348002d39b4SEd Tanous                                      dbus::utility::DbusVariantType>& propPair :
349002d39b4SEd Tanous                      intfPair.second)
350711ac7a9SEd Tanous                 {
351711ac7a9SEd Tanous                     if (propPair.first == "Name")
352711ac7a9SEd Tanous                     {
3535b4aa86bSJames Feist                         const std::string* namePtr =
354711ac7a9SEd Tanous                             std::get_if<std::string>(&propPair.second);
3555b4aa86bSJames Feist                         if (namePtr == nullptr)
3565b4aa86bSJames Feist                         {
3575b4aa86bSJames Feist                             BMCWEB_LOG_ERROR << "Pid Name Field illegal";
358b7a08d04SJames Feist                             messages::internalError(asyncResp->res);
3595b4aa86bSJames Feist                             return;
3605b4aa86bSJames Feist                         }
361db697703SWilly Tu                         name = *namePtr;
3625b4aa86bSJames Feist                         dbus::utility::escapePathForDbus(name);
363711ac7a9SEd Tanous                     }
364711ac7a9SEd Tanous                     else if (propPair.first == "Profiles")
36573df0db0SJames Feist                     {
36673df0db0SJames Feist                         const std::vector<std::string>* profiles =
36773df0db0SJames Feist                             std::get_if<std::vector<std::string>>(
368711ac7a9SEd Tanous                                 &propPair.second);
36973df0db0SJames Feist                         if (profiles == nullptr)
37073df0db0SJames Feist                         {
371002d39b4SEd Tanous                             BMCWEB_LOG_ERROR << "Pid Profiles Field illegal";
37273df0db0SJames Feist                             messages::internalError(asyncResp->res);
37373df0db0SJames Feist                             return;
37473df0db0SJames Feist                         }
37573df0db0SJames Feist                         if (std::find(profiles->begin(), profiles->end(),
37673df0db0SJames Feist                                       currentProfile) == profiles->end())
37773df0db0SJames Feist                         {
37873df0db0SJames Feist                             BMCWEB_LOG_INFO
379002d39b4SEd Tanous                                 << name << " not supported in current profile";
38073df0db0SJames Feist                             continue;
38173df0db0SJames Feist                         }
38273df0db0SJames Feist                     }
383711ac7a9SEd Tanous                 }
384b7a08d04SJames Feist                 nlohmann::json* config = nullptr;
385c33a90ecSJames Feist                 const std::string* classPtr = nullptr;
386711ac7a9SEd Tanous 
387711ac7a9SEd Tanous                 for (const std::pair<std::string,
388002d39b4SEd Tanous                                      dbus::utility::DbusVariantType>& propPair :
389002d39b4SEd Tanous                      intfPair.second)
390c33a90ecSJames Feist                 {
391727dc83fSLei YU                     if (propPair.first == "Class")
392711ac7a9SEd Tanous                     {
393002d39b4SEd Tanous                         classPtr = std::get_if<std::string>(&propPair.second);
394711ac7a9SEd Tanous                     }
395c33a90ecSJames Feist                 }
396c33a90ecSJames Feist 
397eddfc437SWilly Tu                 boost::urls::url url = crow::utility::urlFromPieces(
398eddfc437SWilly Tu                     "redfish", "v1", "Managers", "bmc");
3995b4aa86bSJames Feist                 if (intfPair.first == pidZoneConfigurationIface)
4005b4aa86bSJames Feist                 {
4015b4aa86bSJames Feist                     std::string chassis;
402002d39b4SEd Tanous                     if (!dbus::utility::getNthStringFromPath(pathPair.first.str,
403002d39b4SEd Tanous                                                              5, chassis))
4045b4aa86bSJames Feist                     {
4055b4aa86bSJames Feist                         chassis = "#IllegalValue";
4065b4aa86bSJames Feist                     }
4075b4aa86bSJames Feist                     nlohmann::json& zone = zones[name];
408eddfc437SWilly Tu                     zone["Chassis"]["@odata.id"] = crow::utility::urlFromPieces(
409eddfc437SWilly Tu                         "redfish", "v1", "Chassis", chassis);
410eddfc437SWilly Tu                     url.set_fragment(
411eddfc437SWilly Tu                         ("/Oem/OpenBmc/Fan/FanZones"_json_pointer / name)
412eddfc437SWilly Tu                             .to_string());
413eddfc437SWilly Tu                     zone["@odata.id"] = std::move(url);
4145b4aa86bSJames Feist                     zone["@odata.type"] = "#OemManager.FanZone";
415b7a08d04SJames Feist                     config = &zone;
4165b4aa86bSJames Feist                 }
4175b4aa86bSJames Feist 
418b7a08d04SJames Feist                 else if (intfPair.first == stepwiseConfigurationIface)
4195b4aa86bSJames Feist                 {
420c33a90ecSJames Feist                     if (classPtr == nullptr)
421c33a90ecSJames Feist                     {
422c33a90ecSJames Feist                         BMCWEB_LOG_ERROR << "Pid Class Field illegal";
423c33a90ecSJames Feist                         messages::internalError(asyncResp->res);
424c33a90ecSJames Feist                         return;
425c33a90ecSJames Feist                     }
426c33a90ecSJames Feist 
427b7a08d04SJames Feist                     nlohmann::json& controller = stepwise[name];
428b7a08d04SJames Feist                     config = &controller;
429eddfc437SWilly Tu                     url.set_fragment(
430eddfc437SWilly Tu                         ("/Oem/OpenBmc/Fan/StepwiseControllers"_json_pointer /
431eddfc437SWilly Tu                          name)
432eddfc437SWilly Tu                             .to_string());
433eddfc437SWilly Tu                     controller["@odata.id"] = std::move(url);
434b7a08d04SJames Feist                     controller["@odata.type"] =
435b7a08d04SJames Feist                         "#OemManager.StepwiseController";
436b7a08d04SJames Feist 
437c33a90ecSJames Feist                     controller["Direction"] = *classPtr;
4385b4aa86bSJames Feist                 }
4395b4aa86bSJames Feist 
4405b4aa86bSJames Feist                 // pid and fans are off the same configuration
441b7a08d04SJames Feist                 else if (intfPair.first == pidConfigurationIface)
4425b4aa86bSJames Feist                 {
443c33a90ecSJames Feist 
4445b4aa86bSJames Feist                     if (classPtr == nullptr)
4455b4aa86bSJames Feist                     {
4465b4aa86bSJames Feist                         BMCWEB_LOG_ERROR << "Pid Class Field illegal";
447a08b46ccSJason M. Bills                         messages::internalError(asyncResp->res);
4485b4aa86bSJames Feist                         return;
4495b4aa86bSJames Feist                     }
4505b4aa86bSJames Feist                     bool isFan = *classPtr == "fan";
451002d39b4SEd Tanous                     nlohmann::json& element = isFan ? fans[name] : pids[name];
452b7a08d04SJames Feist                     config = &element;
4535b4aa86bSJames Feist                     if (isFan)
4545b4aa86bSJames Feist                     {
455eddfc437SWilly Tu                         url.set_fragment(
456eddfc437SWilly Tu                             ("/Oem/OpenBmc/Fan/FanControllers"_json_pointer /
457eddfc437SWilly Tu                              name)
458eddfc437SWilly Tu                                 .to_string());
459eddfc437SWilly Tu                         element["@odata.id"] = std::move(url);
460002d39b4SEd Tanous                         element["@odata.type"] = "#OemManager.FanController";
4615b4aa86bSJames Feist                     }
4625b4aa86bSJames Feist                     else
4635b4aa86bSJames Feist                     {
464eddfc437SWilly Tu                         url.set_fragment(
465eddfc437SWilly Tu                             ("/Oem/OpenBmc/Fan/PidControllers"_json_pointer /
466eddfc437SWilly Tu                              name)
467eddfc437SWilly Tu                                 .to_string());
468eddfc437SWilly Tu                         element["@odata.id"] = std::move(url);
469002d39b4SEd Tanous                         element["@odata.type"] = "#OemManager.PidController";
4705b4aa86bSJames Feist                     }
471b7a08d04SJames Feist                 }
472b7a08d04SJames Feist                 else
473b7a08d04SJames Feist                 {
474b7a08d04SJames Feist                     BMCWEB_LOG_ERROR << "Unexpected configuration";
475b7a08d04SJames Feist                     messages::internalError(asyncResp->res);
476b7a08d04SJames Feist                     return;
477b7a08d04SJames Feist                 }
478b7a08d04SJames Feist 
479b7a08d04SJames Feist                 // used for making maps out of 2 vectors
480b7a08d04SJames Feist                 const std::vector<double>* keys = nullptr;
481b7a08d04SJames Feist                 const std::vector<double>* values = nullptr;
482b7a08d04SJames Feist 
483b7a08d04SJames Feist                 for (const auto& propertyPair : intfPair.second)
484b7a08d04SJames Feist                 {
485b7a08d04SJames Feist                     if (propertyPair.first == "Type" ||
486b7a08d04SJames Feist                         propertyPair.first == "Class" ||
487b7a08d04SJames Feist                         propertyPair.first == "Name")
488b7a08d04SJames Feist                     {
489b7a08d04SJames Feist                         continue;
490b7a08d04SJames Feist                     }
491b7a08d04SJames Feist 
492b7a08d04SJames Feist                     // zones
493b7a08d04SJames Feist                     if (intfPair.first == pidZoneConfigurationIface)
494b7a08d04SJames Feist                     {
495b7a08d04SJames Feist                         const double* ptr =
496abf2add6SEd Tanous                             std::get_if<double>(&propertyPair.second);
497b7a08d04SJames Feist                         if (ptr == nullptr)
498b7a08d04SJames Feist                         {
499b7a08d04SJames Feist                             BMCWEB_LOG_ERROR << "Field Illegal "
500b7a08d04SJames Feist                                              << propertyPair.first;
501b7a08d04SJames Feist                             messages::internalError(asyncResp->res);
502b7a08d04SJames Feist                             return;
503b7a08d04SJames Feist                         }
504b7a08d04SJames Feist                         (*config)[propertyPair.first] = *ptr;
505b7a08d04SJames Feist                     }
506b7a08d04SJames Feist 
507b7a08d04SJames Feist                     if (intfPair.first == stepwiseConfigurationIface)
508b7a08d04SJames Feist                     {
509b7a08d04SJames Feist                         if (propertyPair.first == "Reading" ||
510b7a08d04SJames Feist                             propertyPair.first == "Output")
511b7a08d04SJames Feist                         {
512b7a08d04SJames Feist                             const std::vector<double>* ptr =
513abf2add6SEd Tanous                                 std::get_if<std::vector<double>>(
514b7a08d04SJames Feist                                     &propertyPair.second);
515b7a08d04SJames Feist 
516b7a08d04SJames Feist                             if (ptr == nullptr)
517b7a08d04SJames Feist                             {
518b7a08d04SJames Feist                                 BMCWEB_LOG_ERROR << "Field Illegal "
519b7a08d04SJames Feist                                                  << propertyPair.first;
520b7a08d04SJames Feist                                 messages::internalError(asyncResp->res);
521b7a08d04SJames Feist                                 return;
522b7a08d04SJames Feist                             }
523b7a08d04SJames Feist 
524b7a08d04SJames Feist                             if (propertyPair.first == "Reading")
525b7a08d04SJames Feist                             {
526b7a08d04SJames Feist                                 keys = ptr;
527b7a08d04SJames Feist                             }
528b7a08d04SJames Feist                             else
529b7a08d04SJames Feist                             {
530b7a08d04SJames Feist                                 values = ptr;
531b7a08d04SJames Feist                             }
532e662eae8SEd Tanous                             if (keys != nullptr && values != nullptr)
533b7a08d04SJames Feist                             {
534b7a08d04SJames Feist                                 if (keys->size() != values->size())
535b7a08d04SJames Feist                                 {
536b7a08d04SJames Feist                                     BMCWEB_LOG_ERROR
5370fda0f12SGeorge Liu                                         << "Reading and Output size don't match ";
538b7a08d04SJames Feist                                     messages::internalError(asyncResp->res);
539b7a08d04SJames Feist                                     return;
540b7a08d04SJames Feist                                 }
541b7a08d04SJames Feist                                 nlohmann::json& steps = (*config)["Steps"];
542b7a08d04SJames Feist                                 steps = nlohmann::json::array();
543b7a08d04SJames Feist                                 for (size_t ii = 0; ii < keys->size(); ii++)
544b7a08d04SJames Feist                                 {
5451476687dSEd Tanous                                     nlohmann::json::object_t step;
5461476687dSEd Tanous                                     step["Target"] = (*keys)[ii];
5471476687dSEd Tanous                                     step["Output"] = (*values)[ii];
5481476687dSEd Tanous                                     steps.push_back(std::move(step));
549b7a08d04SJames Feist                                 }
550b7a08d04SJames Feist                             }
551b7a08d04SJames Feist                         }
552b7a08d04SJames Feist                         if (propertyPair.first == "NegativeHysteresis" ||
553b7a08d04SJames Feist                             propertyPair.first == "PositiveHysteresis")
554b7a08d04SJames Feist                         {
555b7a08d04SJames Feist                             const double* ptr =
556abf2add6SEd Tanous                                 std::get_if<double>(&propertyPair.second);
557b7a08d04SJames Feist                             if (ptr == nullptr)
558b7a08d04SJames Feist                             {
559b7a08d04SJames Feist                                 BMCWEB_LOG_ERROR << "Field Illegal "
560b7a08d04SJames Feist                                                  << propertyPair.first;
561b7a08d04SJames Feist                                 messages::internalError(asyncResp->res);
562b7a08d04SJames Feist                                 return;
563b7a08d04SJames Feist                             }
564b7a08d04SJames Feist                             (*config)[propertyPair.first] = *ptr;
565b7a08d04SJames Feist                         }
566b7a08d04SJames Feist                     }
567b7a08d04SJames Feist 
568b7a08d04SJames Feist                     // pid and fans are off the same configuration
569b7a08d04SJames Feist                     if (intfPair.first == pidConfigurationIface ||
570b7a08d04SJames Feist                         intfPair.first == stepwiseConfigurationIface)
571b7a08d04SJames Feist                     {
5725b4aa86bSJames Feist 
5735b4aa86bSJames Feist                         if (propertyPair.first == "Zones")
5745b4aa86bSJames Feist                         {
5755b4aa86bSJames Feist                             const std::vector<std::string>* inputs =
576abf2add6SEd Tanous                                 std::get_if<std::vector<std::string>>(
5771b6b96c5SEd Tanous                                     &propertyPair.second);
5785b4aa86bSJames Feist 
5795b4aa86bSJames Feist                             if (inputs == nullptr)
5805b4aa86bSJames Feist                             {
581002d39b4SEd Tanous                                 BMCWEB_LOG_ERROR << "Zones Pid Field Illegal";
582a08b46ccSJason M. Bills                                 messages::internalError(asyncResp->res);
5835b4aa86bSJames Feist                                 return;
5845b4aa86bSJames Feist                             }
585b7a08d04SJames Feist                             auto& data = (*config)[propertyPair.first];
5865b4aa86bSJames Feist                             data = nlohmann::json::array();
5875b4aa86bSJames Feist                             for (std::string itemCopy : *inputs)
5885b4aa86bSJames Feist                             {
5895b4aa86bSJames Feist                                 dbus::utility::escapePathForDbus(itemCopy);
5901476687dSEd Tanous                                 nlohmann::json::object_t input;
591eddfc437SWilly Tu                                 boost::urls::url managerUrl =
592eddfc437SWilly Tu                                     crow::utility::urlFromPieces(
593eddfc437SWilly Tu                                         "redfish", "v1", "Managers", "bmc");
594eddfc437SWilly Tu                                 managerUrl.set_fragment(
595eddfc437SWilly Tu                                     ("/Oem/OpenBmc/Fan/FanZones"_json_pointer /
596eddfc437SWilly Tu                                      itemCopy)
597eddfc437SWilly Tu                                         .to_string());
598eddfc437SWilly Tu                                 input["@odata.id"] = std::move(managerUrl);
5991476687dSEd Tanous                                 data.push_back(std::move(input));
6005b4aa86bSJames Feist                             }
6015b4aa86bSJames Feist                         }
6025b4aa86bSJames Feist                         // todo(james): may never happen, but this
6035b4aa86bSJames Feist                         // assumes configuration data referenced in the
6045b4aa86bSJames Feist                         // PID config is provided by the same daemon, we
6055b4aa86bSJames Feist                         // could add another loop to cover all cases,
6065b4aa86bSJames Feist                         // but I'm okay kicking this can down the road a
6075b4aa86bSJames Feist                         // bit
6085b4aa86bSJames Feist 
6095b4aa86bSJames Feist                         else if (propertyPair.first == "Inputs" ||
6105b4aa86bSJames Feist                                  propertyPair.first == "Outputs")
6115b4aa86bSJames Feist                         {
612b7a08d04SJames Feist                             auto& data = (*config)[propertyPair.first];
6135b4aa86bSJames Feist                             const std::vector<std::string>* inputs =
614abf2add6SEd Tanous                                 std::get_if<std::vector<std::string>>(
6151b6b96c5SEd Tanous                                     &propertyPair.second);
6165b4aa86bSJames Feist 
6175b4aa86bSJames Feist                             if (inputs == nullptr)
6185b4aa86bSJames Feist                             {
6195b4aa86bSJames Feist                                 BMCWEB_LOG_ERROR << "Field Illegal "
6205b4aa86bSJames Feist                                                  << propertyPair.first;
621f12894f8SJason M. Bills                                 messages::internalError(asyncResp->res);
6225b4aa86bSJames Feist                                 return;
6235b4aa86bSJames Feist                             }
6245b4aa86bSJames Feist                             data = *inputs;
625b943aaefSJames Feist                         }
626b943aaefSJames Feist                         else if (propertyPair.first == "SetPointOffset")
627b943aaefSJames Feist                         {
628b943aaefSJames Feist                             const std::string* ptr =
629002d39b4SEd Tanous                                 std::get_if<std::string>(&propertyPair.second);
630b943aaefSJames Feist 
631b943aaefSJames Feist                             if (ptr == nullptr)
632b943aaefSJames Feist                             {
633b943aaefSJames Feist                                 BMCWEB_LOG_ERROR << "Field Illegal "
634b943aaefSJames Feist                                                  << propertyPair.first;
635b943aaefSJames Feist                                 messages::internalError(asyncResp->res);
636b943aaefSJames Feist                                 return;
637b943aaefSJames Feist                             }
638b943aaefSJames Feist                             // translate from dbus to redfish
639b943aaefSJames Feist                             if (*ptr == "WarningHigh")
640b943aaefSJames Feist                             {
641b943aaefSJames Feist                                 (*config)["SetPointOffset"] =
642b943aaefSJames Feist                                     "UpperThresholdNonCritical";
643b943aaefSJames Feist                             }
644b943aaefSJames Feist                             else if (*ptr == "WarningLow")
645b943aaefSJames Feist                             {
646b943aaefSJames Feist                                 (*config)["SetPointOffset"] =
647b943aaefSJames Feist                                     "LowerThresholdNonCritical";
648b943aaefSJames Feist                             }
649b943aaefSJames Feist                             else if (*ptr == "CriticalHigh")
650b943aaefSJames Feist                             {
651b943aaefSJames Feist                                 (*config)["SetPointOffset"] =
652b943aaefSJames Feist                                     "UpperThresholdCritical";
653b943aaefSJames Feist                             }
654b943aaefSJames Feist                             else if (*ptr == "CriticalLow")
655b943aaefSJames Feist                             {
656b943aaefSJames Feist                                 (*config)["SetPointOffset"] =
657b943aaefSJames Feist                                     "LowerThresholdCritical";
658b943aaefSJames Feist                             }
659b943aaefSJames Feist                             else
660b943aaefSJames Feist                             {
661002d39b4SEd Tanous                                 BMCWEB_LOG_ERROR << "Value Illegal " << *ptr;
662b943aaefSJames Feist                                 messages::internalError(asyncResp->res);
663b943aaefSJames Feist                                 return;
664b943aaefSJames Feist                             }
665b943aaefSJames Feist                         }
666b943aaefSJames Feist                         // doubles
667002d39b4SEd Tanous                         else if (propertyPair.first == "FFGainCoefficient" ||
6685b4aa86bSJames Feist                                  propertyPair.first == "FFOffCoefficient" ||
6695b4aa86bSJames Feist                                  propertyPair.first == "ICoefficient" ||
6705b4aa86bSJames Feist                                  propertyPair.first == "ILimitMax" ||
6715b4aa86bSJames Feist                                  propertyPair.first == "ILimitMin" ||
672002d39b4SEd Tanous                                  propertyPair.first == "PositiveHysteresis" ||
673002d39b4SEd Tanous                                  propertyPair.first == "NegativeHysteresis" ||
6745b4aa86bSJames Feist                                  propertyPair.first == "OutLimitMax" ||
6755b4aa86bSJames Feist                                  propertyPair.first == "OutLimitMin" ||
6765b4aa86bSJames Feist                                  propertyPair.first == "PCoefficient" ||
6777625cb81SJames Feist                                  propertyPair.first == "SetPoint" ||
6785b4aa86bSJames Feist                                  propertyPair.first == "SlewNeg" ||
6795b4aa86bSJames Feist                                  propertyPair.first == "SlewPos")
6805b4aa86bSJames Feist                         {
6815b4aa86bSJames Feist                             const double* ptr =
682abf2add6SEd Tanous                                 std::get_if<double>(&propertyPair.second);
6835b4aa86bSJames Feist                             if (ptr == nullptr)
6845b4aa86bSJames Feist                             {
6855b4aa86bSJames Feist                                 BMCWEB_LOG_ERROR << "Field Illegal "
6865b4aa86bSJames Feist                                                  << propertyPair.first;
687f12894f8SJason M. Bills                                 messages::internalError(asyncResp->res);
6885b4aa86bSJames Feist                                 return;
6895b4aa86bSJames Feist                             }
690b7a08d04SJames Feist                             (*config)[propertyPair.first] = *ptr;
6915b4aa86bSJames Feist                         }
6925b4aa86bSJames Feist                     }
6935b4aa86bSJames Feist                 }
6945b4aa86bSJames Feist             }
6955b4aa86bSJames Feist         }
6965b4aa86bSJames Feist         },
6975b4aa86bSJames Feist         connection, path, objectManagerIface, "GetManagedObjects");
6985b4aa86bSJames Feist }
699ca537928SJennifer Lee 
70083ff9ab6SJames Feist enum class CreatePIDRet
70183ff9ab6SJames Feist {
70283ff9ab6SJames Feist     fail,
70383ff9ab6SJames Feist     del,
70483ff9ab6SJames Feist     patch
70583ff9ab6SJames Feist };
70683ff9ab6SJames Feist 
7078d1b46d7Szhanghch05 inline bool
7088d1b46d7Szhanghch05     getZonesFromJsonReq(const std::shared_ptr<bmcweb::AsyncResp>& response,
7095f2caaefSJames Feist                         std::vector<nlohmann::json>& config,
7105f2caaefSJames Feist                         std::vector<std::string>& zones)
7115f2caaefSJames Feist {
712b6baeaa4SJames Feist     if (config.empty())
713b6baeaa4SJames Feist     {
714b6baeaa4SJames Feist         BMCWEB_LOG_ERROR << "Empty Zones";
7151668ce6dSEd Tanous         messages::propertyValueFormatError(response->res, "[]", "Zones");
716b6baeaa4SJames Feist         return false;
717b6baeaa4SJames Feist     }
7185f2caaefSJames Feist     for (auto& odata : config)
7195f2caaefSJames Feist     {
7205f2caaefSJames Feist         std::string path;
7215f2caaefSJames Feist         if (!redfish::json_util::readJson(odata, response->res, "@odata.id",
7225f2caaefSJames Feist                                           path))
7235f2caaefSJames Feist         {
7245f2caaefSJames Feist             return false;
7255f2caaefSJames Feist         }
7265f2caaefSJames Feist         std::string input;
72761adbda3SJames Feist 
72861adbda3SJames Feist         // 8 below comes from
72961adbda3SJames Feist         // /redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/Left
73061adbda3SJames Feist         //     0    1     2      3    4    5      6     7      8
73161adbda3SJames Feist         if (!dbus::utility::getNthStringFromPath(path, 8, input))
7325f2caaefSJames Feist         {
7335f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Got invalid path " << path;
7345f2caaefSJames Feist             BMCWEB_LOG_ERROR << "Illegal Type Zones";
7355f2caaefSJames Feist             messages::propertyValueFormatError(response->res, odata.dump(),
7365f2caaefSJames Feist                                                "Zones");
7375f2caaefSJames Feist             return false;
7385f2caaefSJames Feist         }
739a170f275SEd Tanous         std::replace(input.begin(), input.end(), '_', ' ');
7405f2caaefSJames Feist         zones.emplace_back(std::move(input));
7415f2caaefSJames Feist     }
7425f2caaefSJames Feist     return true;
7435f2caaefSJames Feist }
7445f2caaefSJames Feist 
745711ac7a9SEd Tanous inline const dbus::utility::ManagedObjectType::value_type*
74673df0db0SJames Feist     findChassis(const dbus::utility::ManagedObjectType& managedObj,
747b6baeaa4SJames Feist                 const std::string& value, std::string& chassis)
748b6baeaa4SJames Feist {
749b6baeaa4SJames Feist     BMCWEB_LOG_DEBUG << "Find Chassis: " << value << "\n";
750b6baeaa4SJames Feist 
751a170f275SEd Tanous     std::string escaped = value;
7526ce82fabSYaswanth Reddy M     std::replace(escaped.begin(), escaped.end(), ' ', '_');
753b6baeaa4SJames Feist     escaped = "/" + escaped;
754002d39b4SEd Tanous     auto it = std::find_if(managedObj.begin(), managedObj.end(),
755002d39b4SEd Tanous                            [&escaped](const auto& obj) {
756b6baeaa4SJames Feist         if (boost::algorithm::ends_with(obj.first.str, escaped))
757b6baeaa4SJames Feist         {
758b6baeaa4SJames Feist             BMCWEB_LOG_DEBUG << "Matched " << obj.first.str << "\n";
759b6baeaa4SJames Feist             return true;
760b6baeaa4SJames Feist         }
761b6baeaa4SJames Feist         return false;
762b6baeaa4SJames Feist     });
763b6baeaa4SJames Feist 
764b6baeaa4SJames Feist     if (it == managedObj.end())
765b6baeaa4SJames Feist     {
76673df0db0SJames Feist         return nullptr;
767b6baeaa4SJames Feist     }
768b6baeaa4SJames Feist     // 5 comes from <chassis-name> being the 5th element
769b6baeaa4SJames Feist     // /xyz/openbmc_project/inventory/system/chassis/<chassis-name>
77073df0db0SJames Feist     if (dbus::utility::getNthStringFromPath(it->first.str, 5, chassis))
77173df0db0SJames Feist     {
77273df0db0SJames Feist         return &(*it);
77373df0db0SJames Feist     }
77473df0db0SJames Feist 
77573df0db0SJames Feist     return nullptr;
776b6baeaa4SJames Feist }
777b6baeaa4SJames Feist 
77823a21a1cSEd Tanous inline CreatePIDRet createPidInterface(
7798d1b46d7Szhanghch05     const std::shared_ptr<bmcweb::AsyncResp>& response, const std::string& type,
780b5a76932SEd Tanous     const nlohmann::json::iterator& it, const std::string& path,
78183ff9ab6SJames Feist     const dbus::utility::ManagedObjectType& managedObj, bool createNewObject,
782b9d36b47SEd Tanous     dbus::utility::DBusPropertiesMap& output, std::string& chassis,
783b9d36b47SEd Tanous     const std::string& profile)
78483ff9ab6SJames Feist {
78583ff9ab6SJames Feist 
7865f2caaefSJames Feist     // common deleter
787b6baeaa4SJames Feist     if (it.value() == nullptr)
7885f2caaefSJames Feist     {
7895f2caaefSJames Feist         std::string iface;
7905f2caaefSJames Feist         if (type == "PidControllers" || type == "FanControllers")
7915f2caaefSJames Feist         {
7925f2caaefSJames Feist             iface = pidConfigurationIface;
7935f2caaefSJames Feist         }
7945f2caaefSJames Feist         else if (type == "FanZones")
7955f2caaefSJames Feist         {
7965f2caaefSJames Feist             iface = pidZoneConfigurationIface;
7975f2caaefSJames Feist         }
7985f2caaefSJames Feist         else if (type == "StepwiseControllers")
7995f2caaefSJames Feist         {
8005f2caaefSJames Feist             iface = stepwiseConfigurationIface;
8015f2caaefSJames Feist         }
8025f2caaefSJames Feist         else
8035f2caaefSJames Feist         {
804a0744d38SGunnar Mills             BMCWEB_LOG_ERROR << "Illegal Type " << type;
8055f2caaefSJames Feist             messages::propertyUnknown(response->res, type);
8065f2caaefSJames Feist             return CreatePIDRet::fail;
8075f2caaefSJames Feist         }
8086ee7f774SJames Feist 
8096ee7f774SJames Feist         BMCWEB_LOG_DEBUG << "del " << path << " " << iface << "\n";
8105f2caaefSJames Feist         // delete interface
8115f2caaefSJames Feist         crow::connections::systemBus->async_method_call(
8125e7e2dc5SEd Tanous             [response, path](const boost::system::error_code& ec) {
8135f2caaefSJames Feist             if (ec)
8145f2caaefSJames Feist             {
8155f2caaefSJames Feist                 BMCWEB_LOG_ERROR << "Error patching " << path << ": " << ec;
8165f2caaefSJames Feist                 messages::internalError(response->res);
817b6baeaa4SJames Feist                 return;
8185f2caaefSJames Feist             }
819b6baeaa4SJames Feist             messages::success(response->res);
8205f2caaefSJames Feist             },
8215f2caaefSJames Feist             "xyz.openbmc_project.EntityManager", path, iface, "Delete");
8225f2caaefSJames Feist         return CreatePIDRet::del;
8235f2caaefSJames Feist     }
8245f2caaefSJames Feist 
825711ac7a9SEd Tanous     const dbus::utility::ManagedObjectType::value_type* managedItem = nullptr;
826b6baeaa4SJames Feist     if (!createNewObject)
827b6baeaa4SJames Feist     {
828b6baeaa4SJames Feist         // if we aren't creating a new object, we should be able to find it on
829b6baeaa4SJames Feist         // d-bus
83073df0db0SJames Feist         managedItem = findChassis(managedObj, it.key(), chassis);
83173df0db0SJames Feist         if (managedItem == nullptr)
832b6baeaa4SJames Feist         {
833b6baeaa4SJames Feist             BMCWEB_LOG_ERROR << "Failed to get chassis from config patch";
834ace85d60SEd Tanous             messages::invalidObject(response->res,
835ace85d60SEd Tanous                                     crow::utility::urlFromPieces(
836ace85d60SEd Tanous                                         "redfish", "v1", "Chassis", chassis));
837b6baeaa4SJames Feist             return CreatePIDRet::fail;
838b6baeaa4SJames Feist         }
839b6baeaa4SJames Feist     }
840b6baeaa4SJames Feist 
84126f6976fSEd Tanous     if (!profile.empty() &&
84273df0db0SJames Feist         (type == "PidControllers" || type == "FanControllers" ||
84373df0db0SJames Feist          type == "StepwiseControllers"))
84473df0db0SJames Feist     {
84573df0db0SJames Feist         if (managedItem == nullptr)
84673df0db0SJames Feist         {
847b9d36b47SEd Tanous             output.emplace_back("Profiles", std::vector<std::string>{profile});
84873df0db0SJames Feist         }
84973df0db0SJames Feist         else
85073df0db0SJames Feist         {
85173df0db0SJames Feist             std::string interface;
85273df0db0SJames Feist             if (type == "StepwiseControllers")
85373df0db0SJames Feist             {
85473df0db0SJames Feist                 interface = stepwiseConfigurationIface;
85573df0db0SJames Feist             }
85673df0db0SJames Feist             else
85773df0db0SJames Feist             {
85873df0db0SJames Feist                 interface = pidConfigurationIface;
85973df0db0SJames Feist             }
860711ac7a9SEd Tanous             bool ifaceFound = false;
861711ac7a9SEd Tanous             for (const auto& iface : managedItem->second)
862711ac7a9SEd Tanous             {
863711ac7a9SEd Tanous                 if (iface.first == interface)
864711ac7a9SEd Tanous                 {
865711ac7a9SEd Tanous                     ifaceFound = true;
866711ac7a9SEd Tanous                     for (const auto& prop : iface.second)
867711ac7a9SEd Tanous                     {
868711ac7a9SEd Tanous                         if (prop.first == "Profiles")
869711ac7a9SEd Tanous                         {
870711ac7a9SEd Tanous                             const std::vector<std::string>* curProfiles =
871711ac7a9SEd Tanous                                 std::get_if<std::vector<std::string>>(
872711ac7a9SEd Tanous                                     &(prop.second));
873711ac7a9SEd Tanous                             if (curProfiles == nullptr)
874711ac7a9SEd Tanous                             {
875711ac7a9SEd Tanous                                 BMCWEB_LOG_ERROR
876711ac7a9SEd Tanous                                     << "Illegal profiles in managed object";
877711ac7a9SEd Tanous                                 messages::internalError(response->res);
878711ac7a9SEd Tanous                                 return CreatePIDRet::fail;
879711ac7a9SEd Tanous                             }
880711ac7a9SEd Tanous                             if (std::find(curProfiles->begin(),
881711ac7a9SEd Tanous                                           curProfiles->end(),
882711ac7a9SEd Tanous                                           profile) == curProfiles->end())
883711ac7a9SEd Tanous                             {
884711ac7a9SEd Tanous                                 std::vector<std::string> newProfiles =
885711ac7a9SEd Tanous                                     *curProfiles;
886711ac7a9SEd Tanous                                 newProfiles.push_back(profile);
887b9d36b47SEd Tanous                                 output.emplace_back("Profiles", newProfiles);
888711ac7a9SEd Tanous                             }
889711ac7a9SEd Tanous                         }
890711ac7a9SEd Tanous                     }
891711ac7a9SEd Tanous                 }
892711ac7a9SEd Tanous             }
893711ac7a9SEd Tanous 
894711ac7a9SEd Tanous             if (!ifaceFound)
89573df0db0SJames Feist             {
89673df0db0SJames Feist                 BMCWEB_LOG_ERROR
89773df0db0SJames Feist                     << "Failed to find interface in managed object";
89873df0db0SJames Feist                 messages::internalError(response->res);
89973df0db0SJames Feist                 return CreatePIDRet::fail;
90073df0db0SJames Feist             }
90173df0db0SJames Feist         }
90273df0db0SJames Feist     }
90373df0db0SJames Feist 
90483ff9ab6SJames Feist     if (type == "PidControllers" || type == "FanControllers")
90583ff9ab6SJames Feist     {
90683ff9ab6SJames Feist         if (createNewObject)
90783ff9ab6SJames Feist         {
908b9d36b47SEd Tanous             output.emplace_back("Class",
909b9d36b47SEd Tanous                                 type == "PidControllers" ? "temp" : "fan");
910b9d36b47SEd Tanous             output.emplace_back("Type", "Pid");
91183ff9ab6SJames Feist         }
9125f2caaefSJames Feist 
9135f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> zones;
9145f2caaefSJames Feist         std::optional<std::vector<std::string>> inputs;
9155f2caaefSJames Feist         std::optional<std::vector<std::string>> outputs;
9165f2caaefSJames Feist         std::map<std::string, std::optional<double>> doubles;
917b943aaefSJames Feist         std::optional<std::string> setpointOffset;
9185f2caaefSJames Feist         if (!redfish::json_util::readJson(
919b6baeaa4SJames Feist                 it.value(), response->res, "Inputs", inputs, "Outputs", outputs,
9205f2caaefSJames Feist                 "Zones", zones, "FFGainCoefficient",
9215f2caaefSJames Feist                 doubles["FFGainCoefficient"], "FFOffCoefficient",
9225f2caaefSJames Feist                 doubles["FFOffCoefficient"], "ICoefficient",
9235f2caaefSJames Feist                 doubles["ICoefficient"], "ILimitMax", doubles["ILimitMax"],
9245f2caaefSJames Feist                 "ILimitMin", doubles["ILimitMin"], "OutLimitMax",
9255f2caaefSJames Feist                 doubles["OutLimitMax"], "OutLimitMin", doubles["OutLimitMin"],
9265f2caaefSJames Feist                 "PCoefficient", doubles["PCoefficient"], "SetPoint",
927b943aaefSJames Feist                 doubles["SetPoint"], "SetPointOffset", setpointOffset,
928b943aaefSJames Feist                 "SlewNeg", doubles["SlewNeg"], "SlewPos", doubles["SlewPos"],
929b943aaefSJames Feist                 "PositiveHysteresis", doubles["PositiveHysteresis"],
930b943aaefSJames Feist                 "NegativeHysteresis", doubles["NegativeHysteresis"]))
93183ff9ab6SJames Feist         {
9325f2caaefSJames Feist             return CreatePIDRet::fail;
93383ff9ab6SJames Feist         }
9345f2caaefSJames Feist         if (zones)
9355f2caaefSJames Feist         {
9365f2caaefSJames Feist             std::vector<std::string> zonesStr;
9375f2caaefSJames Feist             if (!getZonesFromJsonReq(response, *zones, zonesStr))
9385f2caaefSJames Feist             {
939a0744d38SGunnar Mills                 BMCWEB_LOG_ERROR << "Illegal Zones";
9405f2caaefSJames Feist                 return CreatePIDRet::fail;
9415f2caaefSJames Feist             }
942b6baeaa4SJames Feist             if (chassis.empty() &&
943e662eae8SEd Tanous                 findChassis(managedObj, zonesStr[0], chassis) == nullptr)
944b6baeaa4SJames Feist             {
945b6baeaa4SJames Feist                 BMCWEB_LOG_ERROR << "Failed to get chassis from config patch";
946ace85d60SEd Tanous                 messages::invalidObject(
947ace85d60SEd Tanous                     response->res, crow::utility::urlFromPieces(
948ace85d60SEd Tanous                                        "redfish", "v1", "Chassis", chassis));
949b6baeaa4SJames Feist                 return CreatePIDRet::fail;
950b6baeaa4SJames Feist             }
951b9d36b47SEd Tanous             output.emplace_back("Zones", std::move(zonesStr));
9525f2caaefSJames Feist         }
953afb9ee06SEd Tanous 
954afb9ee06SEd Tanous         if (inputs)
9555f2caaefSJames Feist         {
956afb9ee06SEd Tanous             for (std::string& value : *inputs)
95783ff9ab6SJames Feist             {
958a170f275SEd Tanous                 std::replace(value.begin(), value.end(), '_', ' ');
95983ff9ab6SJames Feist             }
960afb9ee06SEd Tanous             output.emplace_back("Inputs", *inputs);
961afb9ee06SEd Tanous         }
962afb9ee06SEd Tanous 
963afb9ee06SEd Tanous         if (outputs)
9645f2caaefSJames Feist         {
965afb9ee06SEd Tanous             for (std::string& value : *outputs)
9665f2caaefSJames Feist             {
967afb9ee06SEd Tanous                 std::replace(value.begin(), value.end(), '_', ' ');
9685f2caaefSJames Feist             }
969afb9ee06SEd Tanous             output.emplace_back("Outputs", *outputs);
97083ff9ab6SJames Feist         }
97183ff9ab6SJames Feist 
972b943aaefSJames Feist         if (setpointOffset)
973b943aaefSJames Feist         {
974b943aaefSJames Feist             // translate between redfish and dbus names
975b943aaefSJames Feist             if (*setpointOffset == "UpperThresholdNonCritical")
976b943aaefSJames Feist             {
977b9d36b47SEd Tanous                 output.emplace_back("SetPointOffset", "WarningLow");
978b943aaefSJames Feist             }
979b943aaefSJames Feist             else if (*setpointOffset == "LowerThresholdNonCritical")
980b943aaefSJames Feist             {
981b9d36b47SEd Tanous                 output.emplace_back("SetPointOffset", "WarningHigh");
982b943aaefSJames Feist             }
983b943aaefSJames Feist             else if (*setpointOffset == "LowerThresholdCritical")
984b943aaefSJames Feist             {
985b9d36b47SEd Tanous                 output.emplace_back("SetPointOffset", "CriticalLow");
986b943aaefSJames Feist             }
987b943aaefSJames Feist             else if (*setpointOffset == "UpperThresholdCritical")
988b943aaefSJames Feist             {
989b9d36b47SEd Tanous                 output.emplace_back("SetPointOffset", "CriticalHigh");
990b943aaefSJames Feist             }
991b943aaefSJames Feist             else
992b943aaefSJames Feist             {
993b943aaefSJames Feist                 BMCWEB_LOG_ERROR << "Invalid setpointoffset "
994b943aaefSJames Feist                                  << *setpointOffset;
995ace85d60SEd Tanous                 messages::propertyValueNotInList(response->res, it.key(),
996ace85d60SEd Tanous                                                  "SetPointOffset");
997b943aaefSJames Feist                 return CreatePIDRet::fail;
998b943aaefSJames Feist             }
999b943aaefSJames Feist         }
1000b943aaefSJames Feist 
100183ff9ab6SJames Feist         // doubles
10025f2caaefSJames Feist         for (const auto& pairs : doubles)
100383ff9ab6SJames Feist         {
10045f2caaefSJames Feist             if (!pairs.second)
100583ff9ab6SJames Feist             {
10065f2caaefSJames Feist                 continue;
100783ff9ab6SJames Feist             }
10085f2caaefSJames Feist             BMCWEB_LOG_DEBUG << pairs.first << " = " << *pairs.second;
1009b9d36b47SEd Tanous             output.emplace_back(pairs.first, *pairs.second);
10105f2caaefSJames Feist         }
101183ff9ab6SJames Feist     }
101283ff9ab6SJames Feist 
101383ff9ab6SJames Feist     else if (type == "FanZones")
101483ff9ab6SJames Feist     {
1015b9d36b47SEd Tanous         output.emplace_back("Type", "Pid.Zone");
101683ff9ab6SJames Feist 
10175f2caaefSJames Feist         std::optional<nlohmann::json> chassisContainer;
10185f2caaefSJames Feist         std::optional<double> failSafePercent;
1019d3ec07f8SJames Feist         std::optional<double> minThermalOutput;
1020b6baeaa4SJames Feist         if (!redfish::json_util::readJson(it.value(), response->res, "Chassis",
10215f2caaefSJames Feist                                           chassisContainer, "FailSafePercent",
1022d3ec07f8SJames Feist                                           failSafePercent, "MinThermalOutput",
1023d3ec07f8SJames Feist                                           minThermalOutput))
102483ff9ab6SJames Feist         {
102583ff9ab6SJames Feist             return CreatePIDRet::fail;
102683ff9ab6SJames Feist         }
10275f2caaefSJames Feist 
10285f2caaefSJames Feist         if (chassisContainer)
102983ff9ab6SJames Feist         {
10305f2caaefSJames Feist 
10315f2caaefSJames Feist             std::string chassisId;
10325f2caaefSJames Feist             if (!redfish::json_util::readJson(*chassisContainer, response->res,
10335f2caaefSJames Feist                                               "@odata.id", chassisId))
10345f2caaefSJames Feist             {
103583ff9ab6SJames Feist                 return CreatePIDRet::fail;
103683ff9ab6SJames Feist             }
103783ff9ab6SJames Feist 
1038717794d5SAppaRao Puli             // /redfish/v1/chassis/chassis_name/
10395f2caaefSJames Feist             if (!dbus::utility::getNthStringFromPath(chassisId, 3, chassis))
104083ff9ab6SJames Feist             {
10415f2caaefSJames Feist                 BMCWEB_LOG_ERROR << "Got invalid path " << chassisId;
1042ace85d60SEd Tanous                 messages::invalidObject(
1043ace85d60SEd Tanous                     response->res, crow::utility::urlFromPieces(
1044ace85d60SEd Tanous                                        "redfish", "v1", "Chassis", chassisId));
104583ff9ab6SJames Feist                 return CreatePIDRet::fail;
104683ff9ab6SJames Feist             }
104783ff9ab6SJames Feist         }
1048d3ec07f8SJames Feist         if (minThermalOutput)
104983ff9ab6SJames Feist         {
1050b9d36b47SEd Tanous             output.emplace_back("MinThermalOutput", *minThermalOutput);
10515f2caaefSJames Feist         }
10525f2caaefSJames Feist         if (failSafePercent)
105383ff9ab6SJames Feist         {
1054b9d36b47SEd Tanous             output.emplace_back("FailSafePercent", *failSafePercent);
10555f2caaefSJames Feist         }
10565f2caaefSJames Feist     }
10575f2caaefSJames Feist     else if (type == "StepwiseControllers")
10585f2caaefSJames Feist     {
1059b9d36b47SEd Tanous         output.emplace_back("Type", "Stepwise");
10605f2caaefSJames Feist 
10615f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> zones;
10625f2caaefSJames Feist         std::optional<std::vector<nlohmann::json>> steps;
10635f2caaefSJames Feist         std::optional<std::vector<std::string>> inputs;
10645f2caaefSJames Feist         std::optional<double> positiveHysteresis;
10655f2caaefSJames Feist         std::optional<double> negativeHysteresis;
1066c33a90ecSJames Feist         std::optional<std::string> direction; // upper clipping curve vs lower
10675f2caaefSJames Feist         if (!redfish::json_util::readJson(
1068b6baeaa4SJames Feist                 it.value(), response->res, "Zones", zones, "Steps", steps,
1069b6baeaa4SJames Feist                 "Inputs", inputs, "PositiveHysteresis", positiveHysteresis,
1070c33a90ecSJames Feist                 "NegativeHysteresis", negativeHysteresis, "Direction",
1071c33a90ecSJames Feist                 direction))
10725f2caaefSJames Feist         {
107383ff9ab6SJames Feist             return CreatePIDRet::fail;
107483ff9ab6SJames Feist         }
10755f2caaefSJames Feist 
10765f2caaefSJames Feist         if (zones)
107783ff9ab6SJames Feist         {
1078b6baeaa4SJames Feist             std::vector<std::string> zonesStrs;
1079b6baeaa4SJames Feist             if (!getZonesFromJsonReq(response, *zones, zonesStrs))
10805f2caaefSJames Feist             {
1081a0744d38SGunnar Mills                 BMCWEB_LOG_ERROR << "Illegal Zones";
108283ff9ab6SJames Feist                 return CreatePIDRet::fail;
108383ff9ab6SJames Feist             }
1084b6baeaa4SJames Feist             if (chassis.empty() &&
1085e662eae8SEd Tanous                 findChassis(managedObj, zonesStrs[0], chassis) == nullptr)
1086b6baeaa4SJames Feist             {
1087b6baeaa4SJames Feist                 BMCWEB_LOG_ERROR << "Failed to get chassis from config patch";
1088ace85d60SEd Tanous                 messages::invalidObject(
1089ace85d60SEd Tanous                     response->res, crow::utility::urlFromPieces(
1090ace85d60SEd Tanous                                        "redfish", "v1", "Chassis", chassis));
1091b6baeaa4SJames Feist                 return CreatePIDRet::fail;
1092b6baeaa4SJames Feist             }
1093b9d36b47SEd Tanous             output.emplace_back("Zones", std::move(zonesStrs));
10945f2caaefSJames Feist         }
10955f2caaefSJames Feist         if (steps)
10965f2caaefSJames Feist         {
10975f2caaefSJames Feist             std::vector<double> readings;
10985f2caaefSJames Feist             std::vector<double> outputs;
10995f2caaefSJames Feist             for (auto& step : *steps)
11005f2caaefSJames Feist             {
1101543f4400SEd Tanous                 double target = 0.0;
1102543f4400SEd Tanous                 double out = 0.0;
11035f2caaefSJames Feist 
11045f2caaefSJames Feist                 if (!redfish::json_util::readJson(step, response->res, "Target",
110523a21a1cSEd Tanous                                                   target, "Output", out))
11065f2caaefSJames Feist                 {
11075f2caaefSJames Feist                     return CreatePIDRet::fail;
11085f2caaefSJames Feist                 }
11095f2caaefSJames Feist                 readings.emplace_back(target);
111023a21a1cSEd Tanous                 outputs.emplace_back(out);
11115f2caaefSJames Feist             }
1112b9d36b47SEd Tanous             output.emplace_back("Reading", std::move(readings));
1113b9d36b47SEd Tanous             output.emplace_back("Output", std::move(outputs));
11145f2caaefSJames Feist         }
11155f2caaefSJames Feist         if (inputs)
11165f2caaefSJames Feist         {
11175f2caaefSJames Feist             for (std::string& value : *inputs)
11185f2caaefSJames Feist             {
1119a170f275SEd Tanous 
1120a170f275SEd Tanous                 std::replace(value.begin(), value.end(), '_', ' ');
11215f2caaefSJames Feist             }
1122b9d36b47SEd Tanous             output.emplace_back("Inputs", std::move(*inputs));
11235f2caaefSJames Feist         }
11245f2caaefSJames Feist         if (negativeHysteresis)
11255f2caaefSJames Feist         {
1126b9d36b47SEd Tanous             output.emplace_back("NegativeHysteresis", *negativeHysteresis);
11275f2caaefSJames Feist         }
11285f2caaefSJames Feist         if (positiveHysteresis)
11295f2caaefSJames Feist         {
1130b9d36b47SEd Tanous             output.emplace_back("PositiveHysteresis", *positiveHysteresis);
113183ff9ab6SJames Feist         }
1132c33a90ecSJames Feist         if (direction)
1133c33a90ecSJames Feist         {
1134c33a90ecSJames Feist             constexpr const std::array<const char*, 2> allowedDirections = {
1135c33a90ecSJames Feist                 "Ceiling", "Floor"};
1136c33a90ecSJames Feist             if (std::find(allowedDirections.begin(), allowedDirections.end(),
1137c33a90ecSJames Feist                           *direction) == allowedDirections.end())
1138c33a90ecSJames Feist             {
1139c33a90ecSJames Feist                 messages::propertyValueTypeError(response->res, "Direction",
1140c33a90ecSJames Feist                                                  *direction);
1141c33a90ecSJames Feist                 return CreatePIDRet::fail;
1142c33a90ecSJames Feist             }
1143b9d36b47SEd Tanous             output.emplace_back("Class", *direction);
1144c33a90ecSJames Feist         }
114583ff9ab6SJames Feist     }
114683ff9ab6SJames Feist     else
114783ff9ab6SJames Feist     {
1148a0744d38SGunnar Mills         BMCWEB_LOG_ERROR << "Illegal Type " << type;
114935a62c7cSJason M. Bills         messages::propertyUnknown(response->res, type);
115083ff9ab6SJames Feist         return CreatePIDRet::fail;
115183ff9ab6SJames Feist     }
115283ff9ab6SJames Feist     return CreatePIDRet::patch;
115383ff9ab6SJames Feist }
115473df0db0SJames Feist struct GetPIDValues : std::enable_shared_from_this<GetPIDValues>
115573df0db0SJames Feist {
11566936afe4SEd Tanous     struct CompletionValues
11576936afe4SEd Tanous     {
11586936afe4SEd Tanous         std::vector<std::string> supportedProfiles;
11596936afe4SEd Tanous         std::string currentProfile;
11606936afe4SEd Tanous         dbus::utility::MapperGetSubTreeResponse subtree;
11616936afe4SEd Tanous     };
116283ff9ab6SJames Feist 
11634e23a444SEd Tanous     explicit GetPIDValues(
11644e23a444SEd Tanous         const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn) :
116523a21a1cSEd Tanous         asyncResp(asyncRespIn)
116673df0db0SJames Feist 
11671214b7e7SGunnar Mills     {}
11689c310685SBorawski.Lukasz 
116973df0db0SJames Feist     void run()
11705b4aa86bSJames Feist     {
117173df0db0SJames Feist         std::shared_ptr<GetPIDValues> self = shared_from_this();
117273df0db0SJames Feist 
117373df0db0SJames Feist         // get all configurations
1174e99073f5SGeorge Liu         constexpr std::array<std::string_view, 4> interfaces = {
1175e99073f5SGeorge Liu             pidConfigurationIface, pidZoneConfigurationIface,
1176e99073f5SGeorge Liu             objectManagerIface, stepwiseConfigurationIface};
1177e99073f5SGeorge Liu         dbus::utility::getSubTree(
1178e99073f5SGeorge Liu             "/", 0, interfaces,
1179b9d36b47SEd Tanous             [self](
1180e99073f5SGeorge Liu                 const boost::system::error_code& ec,
1181b9d36b47SEd Tanous                 const dbus::utility::MapperGetSubTreeResponse& subtreeLocal) {
11825b4aa86bSJames Feist             if (ec)
11835b4aa86bSJames Feist             {
11845b4aa86bSJames Feist                 BMCWEB_LOG_ERROR << ec;
118573df0db0SJames Feist                 messages::internalError(self->asyncResp->res);
118673df0db0SJames Feist                 return;
118773df0db0SJames Feist             }
11886936afe4SEd Tanous             self->complete.subtree = subtreeLocal;
1189e99073f5SGeorge Liu             });
119073df0db0SJames Feist 
119173df0db0SJames Feist         // at the same time get the selected profile
1192e99073f5SGeorge Liu         constexpr std::array<std::string_view, 1> thermalModeIfaces = {
1193e99073f5SGeorge Liu             thermalModeIface};
1194e99073f5SGeorge Liu         dbus::utility::getSubTree(
1195e99073f5SGeorge Liu             "/", 0, thermalModeIfaces,
1196b9d36b47SEd Tanous             [self](
1197e99073f5SGeorge Liu                 const boost::system::error_code& ec,
1198b9d36b47SEd Tanous                 const dbus::utility::MapperGetSubTreeResponse& subtreeLocal) {
119923a21a1cSEd Tanous             if (ec || subtreeLocal.empty())
120073df0db0SJames Feist             {
120173df0db0SJames Feist                 return;
120273df0db0SJames Feist             }
120323a21a1cSEd Tanous             if (subtreeLocal[0].second.size() != 1)
120473df0db0SJames Feist             {
120573df0db0SJames Feist                 // invalid mapper response, should never happen
120673df0db0SJames Feist                 BMCWEB_LOG_ERROR << "GetPIDValues: Mapper Error";
120773df0db0SJames Feist                 messages::internalError(self->asyncResp->res);
12085b4aa86bSJames Feist                 return;
12095b4aa86bSJames Feist             }
12105b4aa86bSJames Feist 
121123a21a1cSEd Tanous             const std::string& path = subtreeLocal[0].first;
121223a21a1cSEd Tanous             const std::string& owner = subtreeLocal[0].second[0].first;
1213fac6e53bSKrzysztof Grobelny 
1214fac6e53bSKrzysztof Grobelny             sdbusplus::asio::getAllProperties(
1215fac6e53bSKrzysztof Grobelny                 *crow::connections::systemBus, owner, path, thermalModeIface,
1216168e20c1SEd Tanous                 [path, owner,
12175e7e2dc5SEd Tanous                  self](const boost::system::error_code& ec2,
1218b9d36b47SEd Tanous                        const dbus::utility::DBusPropertiesMap& resp) {
121923a21a1cSEd Tanous                 if (ec2)
122073df0db0SJames Feist                 {
12210fda0f12SGeorge Liu                     BMCWEB_LOG_ERROR
1222002d39b4SEd Tanous                         << "GetPIDValues: Can't get thermalModeIface " << path;
122373df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
122473df0db0SJames Feist                     return;
122573df0db0SJames Feist                 }
1226fac6e53bSKrzysztof Grobelny 
1227271584abSEd Tanous                 const std::string* current = nullptr;
1228271584abSEd Tanous                 const std::vector<std::string>* supported = nullptr;
1229fac6e53bSKrzysztof Grobelny 
1230fac6e53bSKrzysztof Grobelny                 const bool success = sdbusplus::unpackPropertiesNoThrow(
1231fac6e53bSKrzysztof Grobelny                     dbus_utils::UnpackErrorPrinter(), resp, "Current", current,
1232fac6e53bSKrzysztof Grobelny                     "Supported", supported);
1233fac6e53bSKrzysztof Grobelny 
1234fac6e53bSKrzysztof Grobelny                 if (!success)
123573df0db0SJames Feist                 {
1236002d39b4SEd Tanous                     messages::internalError(self->asyncResp->res);
123773df0db0SJames Feist                     return;
123873df0db0SJames Feist                 }
1239fac6e53bSKrzysztof Grobelny 
124073df0db0SJames Feist                 if (current == nullptr || supported == nullptr)
124173df0db0SJames Feist                 {
12420fda0f12SGeorge Liu                     BMCWEB_LOG_ERROR
1243002d39b4SEd Tanous                         << "GetPIDValues: thermal mode iface invalid " << path;
124473df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
124573df0db0SJames Feist                     return;
124673df0db0SJames Feist                 }
12476936afe4SEd Tanous                 self->complete.currentProfile = *current;
12486936afe4SEd Tanous                 self->complete.supportedProfiles = *supported;
1249fac6e53bSKrzysztof Grobelny                 });
1250e99073f5SGeorge Liu             });
125173df0db0SJames Feist     }
125273df0db0SJames Feist 
12536936afe4SEd Tanous     static void
12546936afe4SEd Tanous         processingComplete(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
12556936afe4SEd Tanous                            const CompletionValues& completion)
125673df0db0SJames Feist     {
125773df0db0SJames Feist         if (asyncResp->res.result() != boost::beast::http::status::ok)
125873df0db0SJames Feist         {
125973df0db0SJames Feist             return;
126073df0db0SJames Feist         }
12615b4aa86bSJames Feist         // create map of <connection, path to objMgr>>
12626936afe4SEd Tanous         boost::container::flat_map<
12636936afe4SEd Tanous             std::string, std::string, std::less<>,
12646936afe4SEd Tanous             std::vector<std::pair<std::string, std::string>>>
12656936afe4SEd Tanous             objectMgrPaths;
12666936afe4SEd Tanous         boost::container::flat_set<std::string, std::less<>,
12676936afe4SEd Tanous                                    std::vector<std::string>>
12686936afe4SEd Tanous             calledConnections;
12696936afe4SEd Tanous         for (const auto& pathGroup : completion.subtree)
12705b4aa86bSJames Feist         {
12715b4aa86bSJames Feist             for (const auto& connectionGroup : pathGroup.second)
12725b4aa86bSJames Feist             {
12736bce33bcSJames Feist                 auto findConnection =
12746bce33bcSJames Feist                     calledConnections.find(connectionGroup.first);
12756bce33bcSJames Feist                 if (findConnection != calledConnections.end())
12766bce33bcSJames Feist                 {
12776bce33bcSJames Feist                     break;
12786bce33bcSJames Feist                 }
127973df0db0SJames Feist                 for (const std::string& interface : connectionGroup.second)
12805b4aa86bSJames Feist                 {
12815b4aa86bSJames Feist                     if (interface == objectManagerIface)
12825b4aa86bSJames Feist                     {
128373df0db0SJames Feist                         objectMgrPaths[connectionGroup.first] = pathGroup.first;
12845b4aa86bSJames Feist                     }
12855b4aa86bSJames Feist                     // this list is alphabetical, so we
12865b4aa86bSJames Feist                     // should have found the objMgr by now
12875b4aa86bSJames Feist                     if (interface == pidConfigurationIface ||
1288b7a08d04SJames Feist                         interface == pidZoneConfigurationIface ||
1289b7a08d04SJames Feist                         interface == stepwiseConfigurationIface)
12905b4aa86bSJames Feist                     {
12915b4aa86bSJames Feist                         auto findObjMgr =
12925b4aa86bSJames Feist                             objectMgrPaths.find(connectionGroup.first);
12935b4aa86bSJames Feist                         if (findObjMgr == objectMgrPaths.end())
12945b4aa86bSJames Feist                         {
12955b4aa86bSJames Feist                             BMCWEB_LOG_DEBUG << connectionGroup.first
12965b4aa86bSJames Feist                                              << "Has no Object Manager";
12975b4aa86bSJames Feist                             continue;
12985b4aa86bSJames Feist                         }
12996bce33bcSJames Feist 
13006bce33bcSJames Feist                         calledConnections.insert(connectionGroup.first);
13016bce33bcSJames Feist 
130273df0db0SJames Feist                         asyncPopulatePid(findObjMgr->first, findObjMgr->second,
13036936afe4SEd Tanous                                          completion.currentProfile,
13046936afe4SEd Tanous                                          completion.supportedProfiles,
130573df0db0SJames Feist                                          asyncResp);
13065b4aa86bSJames Feist                         break;
13075b4aa86bSJames Feist                     }
13085b4aa86bSJames Feist                 }
13095b4aa86bSJames Feist             }
13105b4aa86bSJames Feist         }
131173df0db0SJames Feist     }
131273df0db0SJames Feist 
13136936afe4SEd Tanous     ~GetPIDValues()
13146936afe4SEd Tanous     {
13156936afe4SEd Tanous         boost::asio::post(crow::connections::systemBus->get_io_context(),
13166936afe4SEd Tanous                           std::bind_front(&processingComplete, asyncResp,
13176936afe4SEd Tanous                                           std::move(complete)));
13186936afe4SEd Tanous     }
13196936afe4SEd Tanous 
1320ecd6a3a2SEd Tanous     GetPIDValues(const GetPIDValues&) = delete;
1321ecd6a3a2SEd Tanous     GetPIDValues(GetPIDValues&&) = delete;
1322ecd6a3a2SEd Tanous     GetPIDValues& operator=(const GetPIDValues&) = delete;
1323ecd6a3a2SEd Tanous     GetPIDValues& operator=(GetPIDValues&&) = delete;
1324ecd6a3a2SEd Tanous 
13258d1b46d7Szhanghch05     std::shared_ptr<bmcweb::AsyncResp> asyncResp;
13266936afe4SEd Tanous     CompletionValues complete;
132773df0db0SJames Feist };
132873df0db0SJames Feist 
132973df0db0SJames Feist struct SetPIDValues : std::enable_shared_from_this<SetPIDValues>
133073df0db0SJames Feist {
133173df0db0SJames Feist 
13328d1b46d7Szhanghch05     SetPIDValues(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn,
133373df0db0SJames Feist                  nlohmann::json& data) :
1334271584abSEd Tanous         asyncResp(asyncRespIn)
133573df0db0SJames Feist     {
133673df0db0SJames Feist 
133773df0db0SJames Feist         std::optional<nlohmann::json> pidControllers;
133873df0db0SJames Feist         std::optional<nlohmann::json> fanControllers;
133973df0db0SJames Feist         std::optional<nlohmann::json> fanZones;
134073df0db0SJames Feist         std::optional<nlohmann::json> stepwiseControllers;
134173df0db0SJames Feist 
134273df0db0SJames Feist         if (!redfish::json_util::readJson(
134373df0db0SJames Feist                 data, asyncResp->res, "PidControllers", pidControllers,
134473df0db0SJames Feist                 "FanControllers", fanControllers, "FanZones", fanZones,
134573df0db0SJames Feist                 "StepwiseControllers", stepwiseControllers, "Profile", profile))
134673df0db0SJames Feist         {
134773df0db0SJames Feist             return;
134873df0db0SJames Feist         }
134973df0db0SJames Feist         configuration.emplace_back("PidControllers", std::move(pidControllers));
135073df0db0SJames Feist         configuration.emplace_back("FanControllers", std::move(fanControllers));
135173df0db0SJames Feist         configuration.emplace_back("FanZones", std::move(fanZones));
135273df0db0SJames Feist         configuration.emplace_back("StepwiseControllers",
135373df0db0SJames Feist                                    std::move(stepwiseControllers));
135473df0db0SJames Feist     }
1355ecd6a3a2SEd Tanous 
1356ecd6a3a2SEd Tanous     SetPIDValues(const SetPIDValues&) = delete;
1357ecd6a3a2SEd Tanous     SetPIDValues(SetPIDValues&&) = delete;
1358ecd6a3a2SEd Tanous     SetPIDValues& operator=(const SetPIDValues&) = delete;
1359ecd6a3a2SEd Tanous     SetPIDValues& operator=(SetPIDValues&&) = delete;
1360ecd6a3a2SEd Tanous 
136173df0db0SJames Feist     void run()
136273df0db0SJames Feist     {
136373df0db0SJames Feist         if (asyncResp->res.result() != boost::beast::http::status::ok)
136473df0db0SJames Feist         {
136573df0db0SJames Feist             return;
136673df0db0SJames Feist         }
136773df0db0SJames Feist 
136873df0db0SJames Feist         std::shared_ptr<SetPIDValues> self = shared_from_this();
136973df0db0SJames Feist 
137073df0db0SJames Feist         // todo(james): might make sense to do a mapper call here if this
137173df0db0SJames Feist         // interface gets more traction
137273df0db0SJames Feist         crow::connections::systemBus->async_method_call(
13735e7e2dc5SEd Tanous             [self](const boost::system::error_code& ec,
1374914e2d5dSEd Tanous                    const dbus::utility::ManagedObjectType& mObj) {
137573df0db0SJames Feist             if (ec)
137673df0db0SJames Feist             {
137773df0db0SJames Feist                 BMCWEB_LOG_ERROR << "Error communicating to Entity Manager";
137873df0db0SJames Feist                 messages::internalError(self->asyncResp->res);
137973df0db0SJames Feist                 return;
138073df0db0SJames Feist             }
1381e69d9de2SJames Feist             const std::array<const char*, 3> configurations = {
1382e69d9de2SJames Feist                 pidConfigurationIface, pidZoneConfigurationIface,
1383e69d9de2SJames Feist                 stepwiseConfigurationIface};
1384e69d9de2SJames Feist 
138514b0b8d5SJames Feist             for (const auto& [path, object] : mObj)
1386e69d9de2SJames Feist             {
138714b0b8d5SJames Feist                 for (const auto& [interface, _] : object)
1388e69d9de2SJames Feist                 {
1389002d39b4SEd Tanous                     if (std::find(configurations.begin(), configurations.end(),
1390e69d9de2SJames Feist                                   interface) != configurations.end())
1391e69d9de2SJames Feist                     {
139214b0b8d5SJames Feist                         self->objectCount++;
1393e69d9de2SJames Feist                         break;
1394e69d9de2SJames Feist                     }
1395e69d9de2SJames Feist                 }
1396e69d9de2SJames Feist             }
1397914e2d5dSEd Tanous             self->managedObj = mObj;
139873df0db0SJames Feist             },
1399c106b67aSNan Zhou             "xyz.openbmc_project.EntityManager",
1400c106b67aSNan Zhou             "/xyz/openbmc_project/inventory", objectManagerIface,
140173df0db0SJames Feist             "GetManagedObjects");
140273df0db0SJames Feist 
140373df0db0SJames Feist         // at the same time get the profile information
1404e99073f5SGeorge Liu         constexpr std::array<std::string_view, 1> thermalModeIfaces = {
1405e99073f5SGeorge Liu             thermalModeIface};
1406e99073f5SGeorge Liu         dbus::utility::getSubTree(
1407e99073f5SGeorge Liu             "/", 0, thermalModeIfaces,
1408e99073f5SGeorge Liu             [self](const boost::system::error_code& ec,
1409b9d36b47SEd Tanous                    const dbus::utility::MapperGetSubTreeResponse& subtree) {
141073df0db0SJames Feist             if (ec || subtree.empty())
141173df0db0SJames Feist             {
141273df0db0SJames Feist                 return;
141373df0db0SJames Feist             }
141473df0db0SJames Feist             if (subtree[0].second.empty())
141573df0db0SJames Feist             {
141673df0db0SJames Feist                 // invalid mapper response, should never happen
141773df0db0SJames Feist                 BMCWEB_LOG_ERROR << "SetPIDValues: Mapper Error";
141873df0db0SJames Feist                 messages::internalError(self->asyncResp->res);
141973df0db0SJames Feist                 return;
142073df0db0SJames Feist             }
142173df0db0SJames Feist 
142273df0db0SJames Feist             const std::string& path = subtree[0].first;
142373df0db0SJames Feist             const std::string& owner = subtree[0].second[0].first;
1424fac6e53bSKrzysztof Grobelny             sdbusplus::asio::getAllProperties(
1425fac6e53bSKrzysztof Grobelny                 *crow::connections::systemBus, owner, path, thermalModeIface,
14265e7e2dc5SEd Tanous                 [self, path, owner](const boost::system::error_code& ec2,
1427b9d36b47SEd Tanous                                     const dbus::utility::DBusPropertiesMap& r) {
1428cb13a392SEd Tanous                 if (ec2)
142973df0db0SJames Feist                 {
14300fda0f12SGeorge Liu                     BMCWEB_LOG_ERROR
1431002d39b4SEd Tanous                         << "SetPIDValues: Can't get thermalModeIface " << path;
143273df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
143373df0db0SJames Feist                     return;
143473df0db0SJames Feist                 }
1435271584abSEd Tanous                 const std::string* current = nullptr;
1436271584abSEd Tanous                 const std::vector<std::string>* supported = nullptr;
1437fac6e53bSKrzysztof Grobelny 
1438fac6e53bSKrzysztof Grobelny                 const bool success = sdbusplus::unpackPropertiesNoThrow(
1439fac6e53bSKrzysztof Grobelny                     dbus_utils::UnpackErrorPrinter(), r, "Current", current,
1440fac6e53bSKrzysztof Grobelny                     "Supported", supported);
1441fac6e53bSKrzysztof Grobelny 
1442fac6e53bSKrzysztof Grobelny                 if (!success)
144373df0db0SJames Feist                 {
1444002d39b4SEd Tanous                     messages::internalError(self->asyncResp->res);
144573df0db0SJames Feist                     return;
144673df0db0SJames Feist                 }
1447fac6e53bSKrzysztof Grobelny 
144873df0db0SJames Feist                 if (current == nullptr || supported == nullptr)
144973df0db0SJames Feist                 {
14500fda0f12SGeorge Liu                     BMCWEB_LOG_ERROR
1451002d39b4SEd Tanous                         << "SetPIDValues: thermal mode iface invalid " << path;
145273df0db0SJames Feist                     messages::internalError(self->asyncResp->res);
145373df0db0SJames Feist                     return;
145473df0db0SJames Feist                 }
145573df0db0SJames Feist                 self->currentProfile = *current;
145673df0db0SJames Feist                 self->supportedProfiles = *supported;
145773df0db0SJames Feist                 self->profileConnection = owner;
145873df0db0SJames Feist                 self->profilePath = path;
1459fac6e53bSKrzysztof Grobelny                 });
1460e99073f5SGeorge Liu             });
146173df0db0SJames Feist     }
146224b2fe81SEd Tanous     void pidSetDone()
146373df0db0SJames Feist     {
146473df0db0SJames Feist         if (asyncResp->res.result() != boost::beast::http::status::ok)
146573df0db0SJames Feist         {
146673df0db0SJames Feist             return;
14675b4aa86bSJames Feist         }
14688d1b46d7Szhanghch05         std::shared_ptr<bmcweb::AsyncResp> response = asyncResp;
146973df0db0SJames Feist         if (profile)
147073df0db0SJames Feist         {
147173df0db0SJames Feist             if (std::find(supportedProfiles.begin(), supportedProfiles.end(),
147273df0db0SJames Feist                           *profile) == supportedProfiles.end())
147373df0db0SJames Feist             {
147473df0db0SJames Feist                 messages::actionParameterUnknown(response->res, "Profile",
147573df0db0SJames Feist                                                  *profile);
147673df0db0SJames Feist                 return;
147773df0db0SJames Feist             }
147873df0db0SJames Feist             currentProfile = *profile;
147973df0db0SJames Feist             crow::connections::systemBus->async_method_call(
14805e7e2dc5SEd Tanous                 [response](const boost::system::error_code& ec) {
148173df0db0SJames Feist                 if (ec)
148273df0db0SJames Feist                 {
148373df0db0SJames Feist                     BMCWEB_LOG_ERROR << "Error patching profile" << ec;
148473df0db0SJames Feist                     messages::internalError(response->res);
148573df0db0SJames Feist                 }
148673df0db0SJames Feist                 },
148773df0db0SJames Feist                 profileConnection, profilePath,
148873df0db0SJames Feist                 "org.freedesktop.DBus.Properties", "Set", thermalModeIface,
1489168e20c1SEd Tanous                 "Current", dbus::utility::DbusVariantType(*profile));
149073df0db0SJames Feist         }
149173df0db0SJames Feist 
149273df0db0SJames Feist         for (auto& containerPair : configuration)
149373df0db0SJames Feist         {
149473df0db0SJames Feist             auto& container = containerPair.second;
149573df0db0SJames Feist             if (!container)
149673df0db0SJames Feist             {
149773df0db0SJames Feist                 continue;
149873df0db0SJames Feist             }
14996ee7f774SJames Feist             BMCWEB_LOG_DEBUG << *container;
15006ee7f774SJames Feist 
150102cad96eSEd Tanous             const std::string& type = containerPair.first;
150273df0db0SJames Feist 
150373df0db0SJames Feist             for (nlohmann::json::iterator it = container->begin();
150417a897dfSManojkiran Eda                  it != container->end(); ++it)
150573df0db0SJames Feist             {
150673df0db0SJames Feist                 const auto& name = it.key();
1507*cddbf3dfSPotin Lai                 std::string dbusObjName = name;
1508*cddbf3dfSPotin Lai                 std::replace(dbusObjName.begin(), dbusObjName.end(), ' ', '_');
15096ee7f774SJames Feist                 BMCWEB_LOG_DEBUG << "looking for " << name;
15106ee7f774SJames Feist 
151173df0db0SJames Feist                 auto pathItr =
151273df0db0SJames Feist                     std::find_if(managedObj.begin(), managedObj.end(),
1513*cddbf3dfSPotin Lai                                  [&dbusObjName](const auto& obj) {
1514002d39b4SEd Tanous                     return boost::algorithm::ends_with(obj.first.str,
1515*cddbf3dfSPotin Lai                                                        "/" + dbusObjName);
151673df0db0SJames Feist                     });
1517b9d36b47SEd Tanous                 dbus::utility::DBusPropertiesMap output;
151873df0db0SJames Feist 
151973df0db0SJames Feist                 output.reserve(16); // The pid interface length
152073df0db0SJames Feist 
152173df0db0SJames Feist                 // determines if we're patching entity-manager or
152273df0db0SJames Feist                 // creating a new object
152373df0db0SJames Feist                 bool createNewObject = (pathItr == managedObj.end());
15246ee7f774SJames Feist                 BMCWEB_LOG_DEBUG << "Found = " << !createNewObject;
15256ee7f774SJames Feist 
152673df0db0SJames Feist                 std::string iface;
1527ea2b670dSEd Tanous                 if (!createNewObject)
1528ea2b670dSEd Tanous                 {
15298be2b5b6SPotin Lai                     bool findInterface = false;
1530ea2b670dSEd Tanous                     for (const auto& interface : pathItr->second)
1531ea2b670dSEd Tanous                     {
1532ea2b670dSEd Tanous                         if (interface.first == pidConfigurationIface)
1533ea2b670dSEd Tanous                         {
1534ea2b670dSEd Tanous                             if (type == "PidControllers" ||
1535ea2b670dSEd Tanous                                 type == "FanControllers")
153673df0db0SJames Feist                             {
153773df0db0SJames Feist                                 iface = pidConfigurationIface;
15388be2b5b6SPotin Lai                                 findInterface = true;
15398be2b5b6SPotin Lai                                 break;
154073df0db0SJames Feist                             }
154173df0db0SJames Feist                         }
1542ea2b670dSEd Tanous                         else if (interface.first == pidZoneConfigurationIface)
154373df0db0SJames Feist                         {
1544ea2b670dSEd Tanous                             if (type == "FanZones")
154573df0db0SJames Feist                             {
1546ea2b670dSEd Tanous                                 iface = pidConfigurationIface;
15478be2b5b6SPotin Lai                                 findInterface = true;
15488be2b5b6SPotin Lai                                 break;
154973df0db0SJames Feist                             }
155073df0db0SJames Feist                         }
1551ea2b670dSEd Tanous                         else if (interface.first == stepwiseConfigurationIface)
1552ea2b670dSEd Tanous                         {
1553ea2b670dSEd Tanous                             if (type == "StepwiseControllers")
155473df0db0SJames Feist                             {
155573df0db0SJames Feist                                 iface = stepwiseConfigurationIface;
15568be2b5b6SPotin Lai                                 findInterface = true;
15578be2b5b6SPotin Lai                                 break;
15588be2b5b6SPotin Lai                             }
15598be2b5b6SPotin Lai                         }
15608be2b5b6SPotin Lai                     }
15618be2b5b6SPotin Lai 
15628be2b5b6SPotin Lai                     // create new object if interface not found
15638be2b5b6SPotin Lai                     if (!findInterface)
15648be2b5b6SPotin Lai                     {
156573df0db0SJames Feist                         createNewObject = true;
156673df0db0SJames Feist                     }
1567ea2b670dSEd Tanous                 }
15686ee7f774SJames Feist 
15696ee7f774SJames Feist                 if (createNewObject && it.value() == nullptr)
15706ee7f774SJames Feist                 {
15714e0453b1SGunnar Mills                     // can't delete a non-existent object
15721668ce6dSEd Tanous                     messages::propertyValueNotInList(response->res,
15731668ce6dSEd Tanous                                                      it.value().dump(), name);
15746ee7f774SJames Feist                     continue;
15756ee7f774SJames Feist                 }
15766ee7f774SJames Feist 
15776ee7f774SJames Feist                 std::string path;
15786ee7f774SJames Feist                 if (pathItr != managedObj.end())
15796ee7f774SJames Feist                 {
15806ee7f774SJames Feist                     path = pathItr->first.str;
15816ee7f774SJames Feist                 }
15826ee7f774SJames Feist 
158373df0db0SJames Feist                 BMCWEB_LOG_DEBUG << "Create new = " << createNewObject << "\n";
1584e69d9de2SJames Feist 
1585e69d9de2SJames Feist                 // arbitrary limit to avoid attacks
1586e69d9de2SJames Feist                 constexpr const size_t controllerLimit = 500;
158714b0b8d5SJames Feist                 if (createNewObject && objectCount >= controllerLimit)
1588e69d9de2SJames Feist                 {
1589e69d9de2SJames Feist                     messages::resourceExhaustion(response->res, type);
1590e69d9de2SJames Feist                     continue;
1591e69d9de2SJames Feist                 }
1592a170f275SEd Tanous                 std::string escaped = name;
1593a170f275SEd Tanous                 std::replace(escaped.begin(), escaped.end(), '_', ' ');
1594a170f275SEd Tanous                 output.emplace_back("Name", escaped);
159573df0db0SJames Feist 
159673df0db0SJames Feist                 std::string chassis;
159773df0db0SJames Feist                 CreatePIDRet ret = createPidInterface(
15986ee7f774SJames Feist                     response, type, it, path, managedObj, createNewObject,
15996ee7f774SJames Feist                     output, chassis, currentProfile);
160073df0db0SJames Feist                 if (ret == CreatePIDRet::fail)
160173df0db0SJames Feist                 {
160273df0db0SJames Feist                     return;
160373df0db0SJames Feist                 }
16043174e4dfSEd Tanous                 if (ret == CreatePIDRet::del)
160573df0db0SJames Feist                 {
160673df0db0SJames Feist                     continue;
160773df0db0SJames Feist                 }
160873df0db0SJames Feist 
160973df0db0SJames Feist                 if (!createNewObject)
161073df0db0SJames Feist                 {
161173df0db0SJames Feist                     for (const auto& property : output)
161273df0db0SJames Feist                     {
161373df0db0SJames Feist                         crow::connections::systemBus->async_method_call(
161473df0db0SJames Feist                             [response,
161573df0db0SJames Feist                              propertyName{std::string(property.first)}](
16165e7e2dc5SEd Tanous                                 const boost::system::error_code& ec) {
161773df0db0SJames Feist                             if (ec)
161873df0db0SJames Feist                             {
161973df0db0SJames Feist                                 BMCWEB_LOG_ERROR << "Error patching "
1620002d39b4SEd Tanous                                                  << propertyName << ": " << ec;
162173df0db0SJames Feist                                 messages::internalError(response->res);
162273df0db0SJames Feist                                 return;
162373df0db0SJames Feist                             }
162473df0db0SJames Feist                             messages::success(response->res);
162573df0db0SJames Feist                             },
16266ee7f774SJames Feist                             "xyz.openbmc_project.EntityManager", path,
162773df0db0SJames Feist                             "org.freedesktop.DBus.Properties", "Set", iface,
162873df0db0SJames Feist                             property.first, property.second);
162973df0db0SJames Feist                     }
163073df0db0SJames Feist                 }
163173df0db0SJames Feist                 else
163273df0db0SJames Feist                 {
163373df0db0SJames Feist                     if (chassis.empty())
163473df0db0SJames Feist                     {
163573df0db0SJames Feist                         BMCWEB_LOG_ERROR << "Failed to get chassis from config";
1636ace85d60SEd Tanous                         messages::internalError(response->res);
163773df0db0SJames Feist                         return;
163873df0db0SJames Feist                     }
163973df0db0SJames Feist 
164073df0db0SJames Feist                     bool foundChassis = false;
164173df0db0SJames Feist                     for (const auto& obj : managedObj)
164273df0db0SJames Feist                     {
164373df0db0SJames Feist                         if (boost::algorithm::ends_with(obj.first.str, chassis))
164473df0db0SJames Feist                         {
164573df0db0SJames Feist                             chassis = obj.first.str;
164673df0db0SJames Feist                             foundChassis = true;
164773df0db0SJames Feist                             break;
164873df0db0SJames Feist                         }
164973df0db0SJames Feist                     }
165073df0db0SJames Feist                     if (!foundChassis)
165173df0db0SJames Feist                     {
165273df0db0SJames Feist                         BMCWEB_LOG_ERROR << "Failed to find chassis on dbus";
165373df0db0SJames Feist                         messages::resourceMissingAtURI(
1654ace85d60SEd Tanous                             response->res,
1655ace85d60SEd Tanous                             crow::utility::urlFromPieces("redfish", "v1",
1656ace85d60SEd Tanous                                                          "Chassis", chassis));
165773df0db0SJames Feist                         return;
165873df0db0SJames Feist                     }
165973df0db0SJames Feist 
166073df0db0SJames Feist                     crow::connections::systemBus->async_method_call(
16615e7e2dc5SEd Tanous                         [response](const boost::system::error_code& ec) {
166273df0db0SJames Feist                         if (ec)
166373df0db0SJames Feist                         {
166473df0db0SJames Feist                             BMCWEB_LOG_ERROR << "Error Adding Pid Object "
166573df0db0SJames Feist                                              << ec;
166673df0db0SJames Feist                             messages::internalError(response->res);
166773df0db0SJames Feist                             return;
166873df0db0SJames Feist                         }
166973df0db0SJames Feist                         messages::success(response->res);
167073df0db0SJames Feist                         },
167173df0db0SJames Feist                         "xyz.openbmc_project.EntityManager", chassis,
167273df0db0SJames Feist                         "xyz.openbmc_project.AddObject", "AddObject", output);
167373df0db0SJames Feist                 }
167473df0db0SJames Feist             }
167573df0db0SJames Feist         }
167673df0db0SJames Feist     }
167724b2fe81SEd Tanous 
167824b2fe81SEd Tanous     ~SetPIDValues()
167924b2fe81SEd Tanous     {
168024b2fe81SEd Tanous         try
168124b2fe81SEd Tanous         {
168224b2fe81SEd Tanous             pidSetDone();
168324b2fe81SEd Tanous         }
168424b2fe81SEd Tanous         catch (...)
168524b2fe81SEd Tanous         {
168624b2fe81SEd Tanous             BMCWEB_LOG_CRITICAL << "pidSetDone threw exception";
168724b2fe81SEd Tanous         }
168824b2fe81SEd Tanous     }
168924b2fe81SEd Tanous 
16908d1b46d7Szhanghch05     std::shared_ptr<bmcweb::AsyncResp> asyncResp;
169173df0db0SJames Feist     std::vector<std::pair<std::string, std::optional<nlohmann::json>>>
169273df0db0SJames Feist         configuration;
169373df0db0SJames Feist     std::optional<std::string> profile;
169473df0db0SJames Feist     dbus::utility::ManagedObjectType managedObj;
169573df0db0SJames Feist     std::vector<std::string> supportedProfiles;
169673df0db0SJames Feist     std::string currentProfile;
169773df0db0SJames Feist     std::string profileConnection;
169873df0db0SJames Feist     std::string profilePath;
169914b0b8d5SJames Feist     size_t objectCount = 0;
170073df0db0SJames Feist };
170173df0db0SJames Feist 
1702071d8fdfSSunnySrivastava1984 /**
1703071d8fdfSSunnySrivastava1984  * @brief Retrieves BMC manager location data over DBus
1704071d8fdfSSunnySrivastava1984  *
1705071d8fdfSSunnySrivastava1984  * @param[in] aResp Shared pointer for completing asynchronous calls
1706071d8fdfSSunnySrivastava1984  * @param[in] connectionName - service name
1707071d8fdfSSunnySrivastava1984  * @param[in] path - object path
1708071d8fdfSSunnySrivastava1984  * @return none
1709071d8fdfSSunnySrivastava1984  */
17108d1b46d7Szhanghch05 inline void getLocation(const std::shared_ptr<bmcweb::AsyncResp>& aResp,
1711071d8fdfSSunnySrivastava1984                         const std::string& connectionName,
1712071d8fdfSSunnySrivastava1984                         const std::string& path)
1713071d8fdfSSunnySrivastava1984 {
1714071d8fdfSSunnySrivastava1984     BMCWEB_LOG_DEBUG << "Get BMC manager Location data.";
1715071d8fdfSSunnySrivastava1984 
17161e1e598dSJonathan Doman     sdbusplus::asio::getProperty<std::string>(
17171e1e598dSJonathan Doman         *crow::connections::systemBus, connectionName, path,
17181e1e598dSJonathan Doman         "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode",
17195e7e2dc5SEd Tanous         [aResp](const boost::system::error_code& ec,
17201e1e598dSJonathan Doman                 const std::string& property) {
1721071d8fdfSSunnySrivastava1984         if (ec)
1722071d8fdfSSunnySrivastava1984         {
1723071d8fdfSSunnySrivastava1984             BMCWEB_LOG_DEBUG << "DBUS response error for "
1724071d8fdfSSunnySrivastava1984                                 "Location";
1725071d8fdfSSunnySrivastava1984             messages::internalError(aResp->res);
1726071d8fdfSSunnySrivastava1984             return;
1727071d8fdfSSunnySrivastava1984         }
1728071d8fdfSSunnySrivastava1984 
1729071d8fdfSSunnySrivastava1984         aResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] =
17301e1e598dSJonathan Doman             property;
17311e1e598dSJonathan Doman         });
1732071d8fdfSSunnySrivastava1984 }
17337e860f15SJohn Edward Broadbent // avoid name collision systems.hpp
17347e860f15SJohn Edward Broadbent inline void
17357e860f15SJohn Edward Broadbent     managerGetLastResetTime(const std::shared_ptr<bmcweb::AsyncResp>& aResp)
17364bf2b033SGunnar Mills {
17374bf2b033SGunnar Mills     BMCWEB_LOG_DEBUG << "Getting Manager Last Reset Time";
17384bf2b033SGunnar Mills 
17391e1e598dSJonathan Doman     sdbusplus::asio::getProperty<uint64_t>(
17401e1e598dSJonathan Doman         *crow::connections::systemBus, "xyz.openbmc_project.State.BMC",
17411e1e598dSJonathan Doman         "/xyz/openbmc_project/state/bmc0", "xyz.openbmc_project.State.BMC",
17421e1e598dSJonathan Doman         "LastRebootTime",
17435e7e2dc5SEd Tanous         [aResp](const boost::system::error_code& ec,
17441e1e598dSJonathan Doman                 const uint64_t lastResetTime) {
17454bf2b033SGunnar Mills         if (ec)
17464bf2b033SGunnar Mills         {
17474bf2b033SGunnar Mills             BMCWEB_LOG_DEBUG << "D-BUS response error " << ec;
17484bf2b033SGunnar Mills             return;
17494bf2b033SGunnar Mills         }
17504bf2b033SGunnar Mills 
17514bf2b033SGunnar Mills         // LastRebootTime is epoch time, in milliseconds
17524bf2b033SGunnar Mills         // https://github.com/openbmc/phosphor-dbus-interfaces/blob/7f9a128eb9296e926422ddc312c148b625890bb6/xyz/openbmc_project/State/BMC.interface.yaml#L19
17531e1e598dSJonathan Doman         uint64_t lastResetTimeStamp = lastResetTime / 1000;
17544bf2b033SGunnar Mills 
17554bf2b033SGunnar Mills         // Convert to ISO 8601 standard
17564bf2b033SGunnar Mills         aResp->res.jsonValue["LastResetTime"] =
17572b82937eSEd Tanous             redfish::time_utils::getDateTimeUint(lastResetTimeStamp);
17581e1e598dSJonathan Doman         });
17594bf2b033SGunnar Mills }
17604bf2b033SGunnar Mills 
17614bfefa74SGunnar Mills /**
17624bfefa74SGunnar Mills  * @brief Set the running firmware image
17634bfefa74SGunnar Mills  *
17644bfefa74SGunnar Mills  * @param[i,o] aResp - Async response object
17654bfefa74SGunnar Mills  * @param[i] runningFirmwareTarget - Image to make the running image
17664bfefa74SGunnar Mills  *
17674bfefa74SGunnar Mills  * @return void
17684bfefa74SGunnar Mills  */
17697e860f15SJohn Edward Broadbent inline void
17707e860f15SJohn Edward Broadbent     setActiveFirmwareImage(const std::shared_ptr<bmcweb::AsyncResp>& aResp,
1771f23b7296SEd Tanous                            const std::string& runningFirmwareTarget)
17724bfefa74SGunnar Mills {
17734bfefa74SGunnar Mills     // Get the Id from /redfish/v1/UpdateService/FirmwareInventory/<Id>
1774f23b7296SEd Tanous     std::string::size_type idPos = runningFirmwareTarget.rfind('/');
17754bfefa74SGunnar Mills     if (idPos == std::string::npos)
17764bfefa74SGunnar Mills     {
17774bfefa74SGunnar Mills         messages::propertyValueNotInList(aResp->res, runningFirmwareTarget,
17784bfefa74SGunnar Mills                                          "@odata.id");
17794bfefa74SGunnar Mills         BMCWEB_LOG_DEBUG << "Can't parse firmware ID!";
17804bfefa74SGunnar Mills         return;
17814bfefa74SGunnar Mills     }
17824bfefa74SGunnar Mills     idPos++;
17834bfefa74SGunnar Mills     if (idPos >= runningFirmwareTarget.size())
17844bfefa74SGunnar Mills     {
17854bfefa74SGunnar Mills         messages::propertyValueNotInList(aResp->res, runningFirmwareTarget,
17864bfefa74SGunnar Mills                                          "@odata.id");
17874bfefa74SGunnar Mills         BMCWEB_LOG_DEBUG << "Invalid firmware ID.";
17884bfefa74SGunnar Mills         return;
17894bfefa74SGunnar Mills     }
17904bfefa74SGunnar Mills     std::string firmwareId = runningFirmwareTarget.substr(idPos);
17914bfefa74SGunnar Mills 
17924bfefa74SGunnar Mills     // Make sure the image is valid before setting priority
17934bfefa74SGunnar Mills     crow::connections::systemBus->async_method_call(
1794711ac7a9SEd Tanous         [aResp, firmwareId,
17955e7e2dc5SEd Tanous          runningFirmwareTarget](const boost::system::error_code& ec,
1796711ac7a9SEd Tanous                                 dbus::utility::ManagedObjectType& subtree) {
17974bfefa74SGunnar Mills         if (ec)
17984bfefa74SGunnar Mills         {
17994bfefa74SGunnar Mills             BMCWEB_LOG_DEBUG << "D-Bus response error getting objects.";
18004bfefa74SGunnar Mills             messages::internalError(aResp->res);
18014bfefa74SGunnar Mills             return;
18024bfefa74SGunnar Mills         }
18034bfefa74SGunnar Mills 
180426f6976fSEd Tanous         if (subtree.empty())
18054bfefa74SGunnar Mills         {
18064bfefa74SGunnar Mills             BMCWEB_LOG_DEBUG << "Can't find image!";
18074bfefa74SGunnar Mills             messages::internalError(aResp->res);
18084bfefa74SGunnar Mills             return;
18094bfefa74SGunnar Mills         }
18104bfefa74SGunnar Mills 
18114bfefa74SGunnar Mills         bool foundImage = false;
181202cad96eSEd Tanous         for (const auto& object : subtree)
18134bfefa74SGunnar Mills         {
18144bfefa74SGunnar Mills             const std::string& path =
18154bfefa74SGunnar Mills                 static_cast<const std::string&>(object.first);
1816f23b7296SEd Tanous             std::size_t idPos2 = path.rfind('/');
18174bfefa74SGunnar Mills 
18184bfefa74SGunnar Mills             if (idPos2 == std::string::npos)
18194bfefa74SGunnar Mills             {
18204bfefa74SGunnar Mills                 continue;
18214bfefa74SGunnar Mills             }
18224bfefa74SGunnar Mills 
18234bfefa74SGunnar Mills             idPos2++;
18244bfefa74SGunnar Mills             if (idPos2 >= path.size())
18254bfefa74SGunnar Mills             {
18264bfefa74SGunnar Mills                 continue;
18274bfefa74SGunnar Mills             }
18284bfefa74SGunnar Mills 
18294bfefa74SGunnar Mills             if (path.substr(idPos2) == firmwareId)
18304bfefa74SGunnar Mills             {
18314bfefa74SGunnar Mills                 foundImage = true;
18324bfefa74SGunnar Mills                 break;
18334bfefa74SGunnar Mills             }
18344bfefa74SGunnar Mills         }
18354bfefa74SGunnar Mills 
18364bfefa74SGunnar Mills         if (!foundImage)
18374bfefa74SGunnar Mills         {
1838002d39b4SEd Tanous             messages::propertyValueNotInList(aResp->res, runningFirmwareTarget,
1839002d39b4SEd Tanous                                              "@odata.id");
18404bfefa74SGunnar Mills             BMCWEB_LOG_DEBUG << "Invalid firmware ID.";
18414bfefa74SGunnar Mills             return;
18424bfefa74SGunnar Mills         }
18434bfefa74SGunnar Mills 
18448cc8edecSEd Tanous         BMCWEB_LOG_DEBUG << "Setting firmware version " << firmwareId
18458cc8edecSEd Tanous                          << " to priority 0.";
18464bfefa74SGunnar Mills 
18474bfefa74SGunnar Mills         // Only support Immediate
18484bfefa74SGunnar Mills         // An addition could be a Redfish Setting like
18494bfefa74SGunnar Mills         // ActiveSoftwareImageApplyTime and support OnReset
18504bfefa74SGunnar Mills         crow::connections::systemBus->async_method_call(
18515e7e2dc5SEd Tanous             [aResp](const boost::system::error_code& ec2) {
18528a592810SEd Tanous             if (ec2)
18534bfefa74SGunnar Mills             {
18544bfefa74SGunnar Mills                 BMCWEB_LOG_DEBUG << "D-Bus response error setting.";
18554bfefa74SGunnar Mills                 messages::internalError(aResp->res);
18564bfefa74SGunnar Mills                 return;
18574bfefa74SGunnar Mills             }
18584bfefa74SGunnar Mills             doBMCGracefulRestart(aResp);
18594bfefa74SGunnar Mills             },
18604bfefa74SGunnar Mills 
18614bfefa74SGunnar Mills             "xyz.openbmc_project.Software.BMC.Updater",
18624bfefa74SGunnar Mills             "/xyz/openbmc_project/software/" + firmwareId,
18634bfefa74SGunnar Mills             "org.freedesktop.DBus.Properties", "Set",
18647e860f15SJohn Edward Broadbent             "xyz.openbmc_project.Software.RedundancyPriority", "Priority",
1865168e20c1SEd Tanous             dbus::utility::DbusVariantType(static_cast<uint8_t>(0)));
18664bfefa74SGunnar Mills         },
18674bfefa74SGunnar Mills         "xyz.openbmc_project.Software.BMC.Updater",
18687e860f15SJohn Edward Broadbent         "/xyz/openbmc_project/software", "org.freedesktop.DBus.ObjectManager",
18697e860f15SJohn Edward Broadbent         "GetManagedObjects");
18704bfefa74SGunnar Mills }
18714bfefa74SGunnar Mills 
18727e860f15SJohn Edward Broadbent inline void setDateTime(std::shared_ptr<bmcweb::AsyncResp> aResp,
18737e860f15SJohn Edward Broadbent                         std::string datetime)
1874af5d6058SSantosh Puranik {
1875af5d6058SSantosh Puranik     BMCWEB_LOG_DEBUG << "Set date time: " << datetime;
1876af5d6058SSantosh Puranik 
1877c2e32007SEd Tanous     std::optional<redfish::time_utils::usSinceEpoch> us =
1878c2e32007SEd Tanous         redfish::time_utils::dateStringToEpoch(datetime);
1879c2e32007SEd Tanous     if (!us)
1880af5d6058SSantosh Puranik     {
1881c2e32007SEd Tanous         messages::propertyValueFormatError(aResp->res, datetime, "DateTime");
1882c2e32007SEd Tanous         return;
1883c2e32007SEd Tanous     }
1884af5d6058SSantosh Puranik     crow::connections::systemBus->async_method_call(
1885c2e32007SEd Tanous         [aResp{std::move(aResp)},
18865e7e2dc5SEd Tanous          datetime{std::move(datetime)}](const boost::system::error_code& ec) {
1887af5d6058SSantosh Puranik         if (ec)
1888af5d6058SSantosh Puranik         {
1889af5d6058SSantosh Puranik             BMCWEB_LOG_DEBUG << "Failed to set elapsed time. "
1890af5d6058SSantosh Puranik                                 "DBUS response error "
1891af5d6058SSantosh Puranik                              << ec;
1892af5d6058SSantosh Puranik             messages::internalError(aResp->res);
1893af5d6058SSantosh Puranik             return;
1894af5d6058SSantosh Puranik         }
1895af5d6058SSantosh Puranik         aResp->res.jsonValue["DateTime"] = datetime;
1896af5d6058SSantosh Puranik         },
18977e860f15SJohn Edward Broadbent         "xyz.openbmc_project.Time.Manager", "/xyz/openbmc_project/time/bmc",
1898af5d6058SSantosh Puranik         "org.freedesktop.DBus.Properties", "Set",
1899af5d6058SSantosh Puranik         "xyz.openbmc_project.Time.EpochTime", "Elapsed",
1900c2e32007SEd Tanous         dbus::utility::DbusVariantType(us->count()));
190183ff9ab6SJames Feist }
19029c310685SBorawski.Lukasz 
19037e860f15SJohn Edward Broadbent inline void requestRoutesManager(App& app)
19047e860f15SJohn Edward Broadbent {
19057e860f15SJohn Edward Broadbent     std::string uuid = persistent_data::getConfig().systemUuid;
19069c310685SBorawski.Lukasz 
19077e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/")
1908ed398213SEd Tanous         .privileges(redfish::privileges::getManager)
1909002d39b4SEd Tanous         .methods(boost::beast::http::verb::get)(
1910002d39b4SEd Tanous             [&app, uuid](const crow::Request& req,
191145ca1b86SEd Tanous                          const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
19123ba00073SCarson Labrado         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
191345ca1b86SEd Tanous         {
191445ca1b86SEd Tanous             return;
191545ca1b86SEd Tanous         }
19167e860f15SJohn Edward Broadbent         asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc";
1917a51fc2d2SSui Chen         asyncResp->res.jsonValue["@odata.type"] = "#Manager.v1_14_0.Manager";
19187e860f15SJohn Edward Broadbent         asyncResp->res.jsonValue["Id"] = "bmc";
19197e860f15SJohn Edward Broadbent         asyncResp->res.jsonValue["Name"] = "OpenBmc Manager";
19207e860f15SJohn Edward Broadbent         asyncResp->res.jsonValue["Description"] =
19217e860f15SJohn Edward Broadbent             "Baseboard Management Controller";
19227e860f15SJohn Edward Broadbent         asyncResp->res.jsonValue["PowerState"] = "On";
19231476687dSEd Tanous         asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
19241476687dSEd Tanous         asyncResp->res.jsonValue["Status"]["Health"] = "OK";
19251476687dSEd Tanous 
19267e860f15SJohn Edward Broadbent         asyncResp->res.jsonValue["ManagerType"] = "BMC";
19277e860f15SJohn Edward Broadbent         asyncResp->res.jsonValue["UUID"] = systemd_utils::getUuid();
19287e860f15SJohn Edward Broadbent         asyncResp->res.jsonValue["ServiceEntryPointUUID"] = uuid;
1929002d39b4SEd Tanous         asyncResp->res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model
19307e860f15SJohn Edward Broadbent 
19311476687dSEd Tanous         asyncResp->res.jsonValue["LogServices"]["@odata.id"] =
19321476687dSEd Tanous             "/redfish/v1/Managers/bmc/LogServices";
19331476687dSEd Tanous         asyncResp->res.jsonValue["NetworkProtocol"]["@odata.id"] =
19341476687dSEd Tanous             "/redfish/v1/Managers/bmc/NetworkProtocol";
19351476687dSEd Tanous         asyncResp->res.jsonValue["EthernetInterfaces"]["@odata.id"] =
19361476687dSEd Tanous             "/redfish/v1/Managers/bmc/EthernetInterfaces";
19377e860f15SJohn Edward Broadbent 
19387e860f15SJohn Edward Broadbent #ifdef BMCWEB_ENABLE_VM_NBDPROXY
19391476687dSEd Tanous         asyncResp->res.jsonValue["VirtualMedia"]["@odata.id"] =
19401476687dSEd Tanous             "/redfish/v1/Managers/bmc/VirtualMedia";
19417e860f15SJohn Edward Broadbent #endif // BMCWEB_ENABLE_VM_NBDPROXY
19427e860f15SJohn Edward Broadbent 
19437e860f15SJohn Edward Broadbent         // default oem data
19447e860f15SJohn Edward Broadbent         nlohmann::json& oem = asyncResp->res.jsonValue["Oem"];
19457e860f15SJohn Edward Broadbent         nlohmann::json& oemOpenbmc = oem["OpenBmc"];
19467e860f15SJohn Edward Broadbent         oem["@odata.type"] = "#OemManager.Oem";
19477e860f15SJohn Edward Broadbent         oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem";
19487e860f15SJohn Edward Broadbent         oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc";
19497e860f15SJohn Edward Broadbent         oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc";
19501476687dSEd Tanous 
19511476687dSEd Tanous         nlohmann::json::object_t certificates;
19521476687dSEd Tanous         certificates["@odata.id"] =
19531476687dSEd Tanous             "/redfish/v1/Managers/bmc/Truststore/Certificates";
19541476687dSEd Tanous         oemOpenbmc["Certificates"] = std::move(certificates);
19557e860f15SJohn Edward Broadbent 
19567e860f15SJohn Edward Broadbent         // Manager.Reset (an action) can be many values, OpenBMC only
19577e860f15SJohn Edward Broadbent         // supports BMC reboot.
19587e860f15SJohn Edward Broadbent         nlohmann::json& managerReset =
19597e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["Actions"]["#Manager.Reset"];
19607e860f15SJohn Edward Broadbent         managerReset["target"] =
19617e860f15SJohn Edward Broadbent             "/redfish/v1/Managers/bmc/Actions/Manager.Reset";
19627e860f15SJohn Edward Broadbent         managerReset["@Redfish.ActionInfo"] =
19637e860f15SJohn Edward Broadbent             "/redfish/v1/Managers/bmc/ResetActionInfo";
19647e860f15SJohn Edward Broadbent 
19657e860f15SJohn Edward Broadbent         // ResetToDefaults (Factory Reset) has values like
19667e860f15SJohn Edward Broadbent         // PreserveNetworkAndUsers and PreserveNetwork that aren't supported
19677e860f15SJohn Edward Broadbent         // on OpenBMC
19687e860f15SJohn Edward Broadbent         nlohmann::json& resetToDefaults =
19697e860f15SJohn Edward Broadbent             asyncResp->res.jsonValue["Actions"]["#Manager.ResetToDefaults"];
19707e860f15SJohn Edward Broadbent         resetToDefaults["target"] =
19717e860f15SJohn Edward Broadbent             "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults";
1972613dabeaSEd Tanous         resetToDefaults["ResetType@Redfish.AllowableValues"] =
1973613dabeaSEd Tanous             nlohmann::json::array_t({"ResetAll"});
19747e860f15SJohn Edward Broadbent 
19757c8c4058STejas Patil         std::pair<std::string, std::string> redfishDateTimeOffset =
19762b82937eSEd Tanous             redfish::time_utils::getDateTimeOffsetNow();
19777c8c4058STejas Patil 
19787c8c4058STejas Patil         asyncResp->res.jsonValue["DateTime"] = redfishDateTimeOffset.first;
19797c8c4058STejas Patil         asyncResp->res.jsonValue["DateTimeLocalOffset"] =
19807c8c4058STejas Patil             redfishDateTimeOffset.second;
19817e860f15SJohn Edward Broadbent 
19820e8ac5e7SGunnar Mills         // TODO (Gunnar): Remove these one day since moved to ComputerSystem
19830e8ac5e7SGunnar Mills         // Still used by OCP profiles
19840e8ac5e7SGunnar Mills         // https://github.com/opencomputeproject/OCP-Profiles/issues/23
19857e860f15SJohn Edward Broadbent         // Fill in SerialConsole info
19867e860f15SJohn Edward Broadbent         asyncResp->res.jsonValue["SerialConsole"]["ServiceEnabled"] = true;
1987002d39b4SEd Tanous         asyncResp->res.jsonValue["SerialConsole"]["MaxConcurrentSessions"] = 15;
1988613dabeaSEd Tanous         asyncResp->res.jsonValue["SerialConsole"]["ConnectTypesSupported"] =
1989613dabeaSEd Tanous             nlohmann::json::array_t({"IPMI", "SSH"});
19907e860f15SJohn Edward Broadbent #ifdef BMCWEB_ENABLE_KVM
19917e860f15SJohn Edward Broadbent         // Fill in GraphicalConsole info
1992002d39b4SEd Tanous         asyncResp->res.jsonValue["GraphicalConsole"]["ServiceEnabled"] = true;
1993002d39b4SEd Tanous         asyncResp->res.jsonValue["GraphicalConsole"]["MaxConcurrentSessions"] =
1994002d39b4SEd Tanous             4;
1995613dabeaSEd Tanous         asyncResp->res.jsonValue["GraphicalConsole"]["ConnectTypesSupported"] =
1996613dabeaSEd Tanous             nlohmann::json::array_t({"KVMIP"});
19977e860f15SJohn Edward Broadbent #endif // BMCWEB_ENABLE_KVM
19987e860f15SJohn Edward Broadbent 
1999002d39b4SEd Tanous         asyncResp->res.jsonValue["Links"]["ManagerForServers@odata.count"] = 1;
20001476687dSEd Tanous 
20011476687dSEd Tanous         nlohmann::json::array_t managerForServers;
20021476687dSEd Tanous         nlohmann::json::object_t manager;
20031476687dSEd Tanous         manager["@odata.id"] = "/redfish/v1/Systems/system";
20041476687dSEd Tanous         managerForServers.push_back(std::move(manager));
20051476687dSEd Tanous 
20061476687dSEd Tanous         asyncResp->res.jsonValue["Links"]["ManagerForServers"] =
20071476687dSEd Tanous             std::move(managerForServers);
20087e860f15SJohn Edward Broadbent 
20097e860f15SJohn Edward Broadbent         auto health = std::make_shared<HealthPopulate>(asyncResp);
20107e860f15SJohn Edward Broadbent         health->isManagersHealth = true;
20117e860f15SJohn Edward Broadbent         health->populate();
20127e860f15SJohn Edward Broadbent 
2013eee0013eSWilly Tu         sw_util::populateSoftwareInformation(asyncResp, sw_util::bmcPurpose,
20147e860f15SJohn Edward Broadbent                                              "FirmwareVersion", true);
20157e860f15SJohn Edward Broadbent 
20167e860f15SJohn Edward Broadbent         managerGetLastResetTime(asyncResp);
20177e860f15SJohn Edward Broadbent 
2018a51fc2d2SSui Chen         // ManagerDiagnosticData is added for all BMCs.
2019a51fc2d2SSui Chen         nlohmann::json& managerDiagnosticData =
2020a51fc2d2SSui Chen             asyncResp->res.jsonValue["ManagerDiagnosticData"];
2021a51fc2d2SSui Chen         managerDiagnosticData["@odata.id"] =
2022a51fc2d2SSui Chen             "/redfish/v1/Managers/bmc/ManagerDiagnosticData";
2023a51fc2d2SSui Chen 
202454dce7f5SGunnar Mills #ifdef BMCWEB_ENABLE_REDFISH_OEM_MANAGER_FAN_DATA
20257e860f15SJohn Edward Broadbent         auto pids = std::make_shared<GetPIDValues>(asyncResp);
20267e860f15SJohn Edward Broadbent         pids->run();
202754dce7f5SGunnar Mills #endif
20287e860f15SJohn Edward Broadbent 
2029002d39b4SEd Tanous         getMainChassisId(asyncResp,
2030002d39b4SEd Tanous                          [](const std::string& chassisId,
2031002d39b4SEd Tanous                             const std::shared_ptr<bmcweb::AsyncResp>& aRsp) {
2032002d39b4SEd Tanous             aRsp->res.jsonValue["Links"]["ManagerForChassis@odata.count"] = 1;
20331476687dSEd Tanous             nlohmann::json::array_t managerForChassis;
20348a592810SEd Tanous             nlohmann::json::object_t managerObj;
2035eddfc437SWilly Tu             boost::urls::url chassiUrl = crow::utility::urlFromPieces(
2036eddfc437SWilly Tu                 "redfish", "v1", "Chassis", chassisId);
2037eddfc437SWilly Tu             managerObj["@odata.id"] = chassiUrl;
20388a592810SEd Tanous             managerForChassis.push_back(std::move(managerObj));
20391476687dSEd Tanous             aRsp->res.jsonValue["Links"]["ManagerForChassis"] =
20401476687dSEd Tanous                 std::move(managerForChassis);
20411476687dSEd Tanous             aRsp->res.jsonValue["Links"]["ManagerInChassis"]["@odata.id"] =
2042eddfc437SWilly Tu                 chassiUrl;
20437e860f15SJohn Edward Broadbent         });
20447e860f15SJohn Edward Broadbent 
20457e860f15SJohn Edward Broadbent         static bool started = false;
20467e860f15SJohn Edward Broadbent 
20477e860f15SJohn Edward Broadbent         if (!started)
20481abe55efSEd Tanous         {
20491e1e598dSJonathan Doman             sdbusplus::asio::getProperty<double>(
20501e1e598dSJonathan Doman                 *crow::connections::systemBus, "org.freedesktop.systemd1",
2051002d39b4SEd Tanous                 "/org/freedesktop/systemd1", "org.freedesktop.systemd1.Manager",
2052002d39b4SEd Tanous                 "Progress",
20535e7e2dc5SEd Tanous                 [asyncResp](const boost::system::error_code& ec,
20541e1e598dSJonathan Doman                             const double& val) {
20557e860f15SJohn Edward Broadbent                 if (ec)
20561abe55efSEd Tanous                 {
20577e860f15SJohn Edward Broadbent                     BMCWEB_LOG_ERROR << "Error while getting progress";
20587e860f15SJohn Edward Broadbent                     messages::internalError(asyncResp->res);
20597e860f15SJohn Edward Broadbent                     return;
20607e860f15SJohn Edward Broadbent                 }
20611e1e598dSJonathan Doman                 if (val < 1.0)
20627e860f15SJohn Edward Broadbent                 {
2063002d39b4SEd Tanous                     asyncResp->res.jsonValue["Status"]["State"] = "Starting";
20647e860f15SJohn Edward Broadbent                     started = true;
20657e860f15SJohn Edward Broadbent                 }
20661e1e598dSJonathan Doman                 });
20679c310685SBorawski.Lukasz         }
20689c310685SBorawski.Lukasz 
2069e99073f5SGeorge Liu         constexpr std::array<std::string_view, 1> interfaces = {
2070e99073f5SGeorge Liu             "xyz.openbmc_project.Inventory.Item.Bmc"};
2071e99073f5SGeorge Liu         dbus::utility::getSubTree(
2072e99073f5SGeorge Liu             "/xyz/openbmc_project/inventory", 0, interfaces,
20737e860f15SJohn Edward Broadbent             [asyncResp](
2074e99073f5SGeorge Liu                 const boost::system::error_code& ec,
2075b9d36b47SEd Tanous                 const dbus::utility::MapperGetSubTreeResponse& subtree) {
20767e860f15SJohn Edward Broadbent             if (ec)
20771abe55efSEd Tanous             {
2078002d39b4SEd Tanous                 BMCWEB_LOG_DEBUG << "D-Bus response error on GetSubTree " << ec;
20797e860f15SJohn Edward Broadbent                 return;
20807e860f15SJohn Edward Broadbent             }
208126f6976fSEd Tanous             if (subtree.empty())
20827e860f15SJohn Edward Broadbent             {
20837e860f15SJohn Edward Broadbent                 BMCWEB_LOG_DEBUG << "Can't find bmc D-Bus object!";
20847e860f15SJohn Edward Broadbent                 return;
20857e860f15SJohn Edward Broadbent             }
20867e860f15SJohn Edward Broadbent             // Assume only 1 bmc D-Bus object
20877e860f15SJohn Edward Broadbent             // Throw an error if there is more than 1
20887e860f15SJohn Edward Broadbent             if (subtree.size() > 1)
20897e860f15SJohn Edward Broadbent             {
2090002d39b4SEd Tanous                 BMCWEB_LOG_DEBUG << "Found more than 1 bmc D-Bus object!";
20917e860f15SJohn Edward Broadbent                 messages::internalError(asyncResp->res);
20927e860f15SJohn Edward Broadbent                 return;
20937e860f15SJohn Edward Broadbent             }
20947e860f15SJohn Edward Broadbent 
2095002d39b4SEd Tanous             if (subtree[0].first.empty() || subtree[0].second.size() != 1)
20967e860f15SJohn Edward Broadbent             {
20977e860f15SJohn Edward Broadbent                 BMCWEB_LOG_DEBUG << "Error getting bmc D-Bus object!";
20987e860f15SJohn Edward Broadbent                 messages::internalError(asyncResp->res);
20997e860f15SJohn Edward Broadbent                 return;
21007e860f15SJohn Edward Broadbent             }
21017e860f15SJohn Edward Broadbent 
21027e860f15SJohn Edward Broadbent             const std::string& path = subtree[0].first;
2103002d39b4SEd Tanous             const std::string& connectionName = subtree[0].second[0].first;
21047e860f15SJohn Edward Broadbent 
2105002d39b4SEd Tanous             for (const auto& interfaceName : subtree[0].second[0].second)
21067e860f15SJohn Edward Broadbent             {
21077e860f15SJohn Edward Broadbent                 if (interfaceName ==
21087e860f15SJohn Edward Broadbent                     "xyz.openbmc_project.Inventory.Decorator.Asset")
21097e860f15SJohn Edward Broadbent                 {
2110fac6e53bSKrzysztof Grobelny 
2111fac6e53bSKrzysztof Grobelny                     sdbusplus::asio::getAllProperties(
2112fac6e53bSKrzysztof Grobelny                         *crow::connections::systemBus, connectionName, path,
2113fac6e53bSKrzysztof Grobelny                         "xyz.openbmc_project.Inventory.Decorator.Asset",
21145e7e2dc5SEd Tanous                         [asyncResp](const boost::system::error_code& ec2,
2115b9d36b47SEd Tanous                                     const dbus::utility::DBusPropertiesMap&
21167e860f15SJohn Edward Broadbent                                         propertiesList) {
21178a592810SEd Tanous                         if (ec2)
21187e860f15SJohn Edward Broadbent                         {
2119002d39b4SEd Tanous                             BMCWEB_LOG_DEBUG << "Can't get bmc asset!";
21207e860f15SJohn Edward Broadbent                             return;
21217e860f15SJohn Edward Broadbent                         }
21227e860f15SJohn Edward Broadbent 
2123fac6e53bSKrzysztof Grobelny                         const std::string* partNumber = nullptr;
2124fac6e53bSKrzysztof Grobelny                         const std::string* serialNumber = nullptr;
2125fac6e53bSKrzysztof Grobelny                         const std::string* manufacturer = nullptr;
2126fac6e53bSKrzysztof Grobelny                         const std::string* model = nullptr;
2127fac6e53bSKrzysztof Grobelny                         const std::string* sparePartNumber = nullptr;
2128fac6e53bSKrzysztof Grobelny 
2129fac6e53bSKrzysztof Grobelny                         const bool success = sdbusplus::unpackPropertiesNoThrow(
2130fac6e53bSKrzysztof Grobelny                             dbus_utils::UnpackErrorPrinter(), propertiesList,
2131fac6e53bSKrzysztof Grobelny                             "PartNumber", partNumber, "SerialNumber",
2132fac6e53bSKrzysztof Grobelny                             serialNumber, "Manufacturer", manufacturer, "Model",
2133fac6e53bSKrzysztof Grobelny                             model, "SparePartNumber", sparePartNumber);
2134fac6e53bSKrzysztof Grobelny 
2135fac6e53bSKrzysztof Grobelny                         if (!success)
21367e860f15SJohn Edward Broadbent                         {
2137002d39b4SEd Tanous                             messages::internalError(asyncResp->res);
21387e860f15SJohn Edward Broadbent                             return;
21397e860f15SJohn Edward Broadbent                         }
2140fac6e53bSKrzysztof Grobelny 
2141fac6e53bSKrzysztof Grobelny                         if (partNumber != nullptr)
2142fac6e53bSKrzysztof Grobelny                         {
2143fac6e53bSKrzysztof Grobelny                             asyncResp->res.jsonValue["PartNumber"] =
2144fac6e53bSKrzysztof Grobelny                                 *partNumber;
21457e860f15SJohn Edward Broadbent                         }
2146fac6e53bSKrzysztof Grobelny 
2147fac6e53bSKrzysztof Grobelny                         if (serialNumber != nullptr)
2148fac6e53bSKrzysztof Grobelny                         {
2149fac6e53bSKrzysztof Grobelny                             asyncResp->res.jsonValue["SerialNumber"] =
2150fac6e53bSKrzysztof Grobelny                                 *serialNumber;
21517e860f15SJohn Edward Broadbent                         }
2152fac6e53bSKrzysztof Grobelny 
2153fac6e53bSKrzysztof Grobelny                         if (manufacturer != nullptr)
2154fac6e53bSKrzysztof Grobelny                         {
2155fac6e53bSKrzysztof Grobelny                             asyncResp->res.jsonValue["Manufacturer"] =
2156fac6e53bSKrzysztof Grobelny                                 *manufacturer;
2157fac6e53bSKrzysztof Grobelny                         }
2158fac6e53bSKrzysztof Grobelny 
2159fac6e53bSKrzysztof Grobelny                         if (model != nullptr)
2160fac6e53bSKrzysztof Grobelny                         {
2161fac6e53bSKrzysztof Grobelny                             asyncResp->res.jsonValue["Model"] = *model;
2162fac6e53bSKrzysztof Grobelny                         }
2163fac6e53bSKrzysztof Grobelny 
2164fac6e53bSKrzysztof Grobelny                         if (sparePartNumber != nullptr)
2165fac6e53bSKrzysztof Grobelny                         {
2166fac6e53bSKrzysztof Grobelny                             asyncResp->res.jsonValue["SparePartNumber"] =
2167fac6e53bSKrzysztof Grobelny                                 *sparePartNumber;
2168fac6e53bSKrzysztof Grobelny                         }
2169fac6e53bSKrzysztof Grobelny                         });
21707e860f15SJohn Edward Broadbent                 }
2171002d39b4SEd Tanous                 else if (interfaceName ==
21720fda0f12SGeorge Liu                          "xyz.openbmc_project.Inventory.Decorator.LocationCode")
21737e860f15SJohn Edward Broadbent                 {
21747e860f15SJohn Edward Broadbent                     getLocation(asyncResp, connectionName, path);
21757e860f15SJohn Edward Broadbent                 }
21767e860f15SJohn Edward Broadbent             }
2177e99073f5SGeorge Liu             });
21787e860f15SJohn Edward Broadbent         });
21797e860f15SJohn Edward Broadbent 
21807e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/")
2181ed398213SEd Tanous         .privileges(redfish::privileges::patchManager)
218245ca1b86SEd Tanous         .methods(boost::beast::http::verb::patch)(
218345ca1b86SEd Tanous             [&app](const crow::Request& req,
21847e860f15SJohn Edward Broadbent                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
21853ba00073SCarson Labrado         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
218645ca1b86SEd Tanous         {
218745ca1b86SEd Tanous             return;
218845ca1b86SEd Tanous         }
21897e860f15SJohn Edward Broadbent         std::optional<nlohmann::json> oem;
21907e860f15SJohn Edward Broadbent         std::optional<nlohmann::json> links;
21917e860f15SJohn Edward Broadbent         std::optional<std::string> datetime;
21927e860f15SJohn Edward Broadbent 
219315ed6780SWilly Tu         if (!json_util::readJsonPatch(req, asyncResp->res, "Oem", oem,
2194002d39b4SEd Tanous                                       "DateTime", datetime, "Links", links))
21957e860f15SJohn Edward Broadbent         {
21967e860f15SJohn Edward Broadbent             return;
21977e860f15SJohn Edward Broadbent         }
21987e860f15SJohn Edward Broadbent 
21997e860f15SJohn Edward Broadbent         if (oem)
22007e860f15SJohn Edward Broadbent         {
220154dce7f5SGunnar Mills #ifdef BMCWEB_ENABLE_REDFISH_OEM_MANAGER_FAN_DATA
22027e860f15SJohn Edward Broadbent             std::optional<nlohmann::json> openbmc;
2203002d39b4SEd Tanous             if (!redfish::json_util::readJson(*oem, asyncResp->res, "OpenBmc",
2204002d39b4SEd Tanous                                               openbmc))
22057e860f15SJohn Edward Broadbent             {
22067e860f15SJohn Edward Broadbent                 return;
22077e860f15SJohn Edward Broadbent             }
22087e860f15SJohn Edward Broadbent             if (openbmc)
22097e860f15SJohn Edward Broadbent             {
22107e860f15SJohn Edward Broadbent                 std::optional<nlohmann::json> fan;
2211002d39b4SEd Tanous                 if (!redfish::json_util::readJson(*openbmc, asyncResp->res,
2212002d39b4SEd Tanous                                                   "Fan", fan))
22137e860f15SJohn Edward Broadbent                 {
22147e860f15SJohn Edward Broadbent                     return;
22157e860f15SJohn Edward Broadbent                 }
22167e860f15SJohn Edward Broadbent                 if (fan)
22177e860f15SJohn Edward Broadbent                 {
2218002d39b4SEd Tanous                     auto pid = std::make_shared<SetPIDValues>(asyncResp, *fan);
22197e860f15SJohn Edward Broadbent                     pid->run();
22207e860f15SJohn Edward Broadbent                 }
22217e860f15SJohn Edward Broadbent             }
222254dce7f5SGunnar Mills #else
222354dce7f5SGunnar Mills             messages::propertyUnknown(asyncResp->res, "Oem");
222454dce7f5SGunnar Mills             return;
222554dce7f5SGunnar Mills #endif
22267e860f15SJohn Edward Broadbent         }
22277e860f15SJohn Edward Broadbent         if (links)
22287e860f15SJohn Edward Broadbent         {
22297e860f15SJohn Edward Broadbent             std::optional<nlohmann::json> activeSoftwareImage;
22307e860f15SJohn Edward Broadbent             if (!redfish::json_util::readJson(*links, asyncResp->res,
22317e860f15SJohn Edward Broadbent                                               "ActiveSoftwareImage",
22327e860f15SJohn Edward Broadbent                                               activeSoftwareImage))
22337e860f15SJohn Edward Broadbent             {
22347e860f15SJohn Edward Broadbent                 return;
22357e860f15SJohn Edward Broadbent             }
22367e860f15SJohn Edward Broadbent             if (activeSoftwareImage)
22377e860f15SJohn Edward Broadbent             {
22387e860f15SJohn Edward Broadbent                 std::optional<std::string> odataId;
2239002d39b4SEd Tanous                 if (!json_util::readJson(*activeSoftwareImage, asyncResp->res,
2240002d39b4SEd Tanous                                          "@odata.id", odataId))
22417e860f15SJohn Edward Broadbent                 {
22427e860f15SJohn Edward Broadbent                     return;
22437e860f15SJohn Edward Broadbent                 }
22447e860f15SJohn Edward Broadbent 
22457e860f15SJohn Edward Broadbent                 if (odataId)
22467e860f15SJohn Edward Broadbent                 {
22477e860f15SJohn Edward Broadbent                     setActiveFirmwareImage(asyncResp, *odataId);
22487e860f15SJohn Edward Broadbent                 }
22497e860f15SJohn Edward Broadbent             }
22507e860f15SJohn Edward Broadbent         }
22517e860f15SJohn Edward Broadbent         if (datetime)
22527e860f15SJohn Edward Broadbent         {
22537e860f15SJohn Edward Broadbent             setDateTime(asyncResp, std::move(*datetime));
22547e860f15SJohn Edward Broadbent         }
22557e860f15SJohn Edward Broadbent         });
22567e860f15SJohn Edward Broadbent }
22577e860f15SJohn Edward Broadbent 
22587e860f15SJohn Edward Broadbent inline void requestRoutesManagerCollection(App& app)
22597e860f15SJohn Edward Broadbent {
22607e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app, "/redfish/v1/Managers/")
2261ed398213SEd Tanous         .privileges(redfish::privileges::getManagerCollection)
22627e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::get)(
226345ca1b86SEd Tanous             [&app](const crow::Request& req,
22647e860f15SJohn Edward Broadbent                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
22653ba00073SCarson Labrado         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
226645ca1b86SEd Tanous         {
226745ca1b86SEd Tanous             return;
226845ca1b86SEd Tanous         }
226983ff9ab6SJames Feist         // Collections don't include the static data added by SubRoute
227083ff9ab6SJames Feist         // because it has a duplicate entry for members
22718d1b46d7Szhanghch05         asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers";
22728d1b46d7Szhanghch05         asyncResp->res.jsonValue["@odata.type"] =
22738d1b46d7Szhanghch05             "#ManagerCollection.ManagerCollection";
22748d1b46d7Szhanghch05         asyncResp->res.jsonValue["Name"] = "Manager Collection";
22758d1b46d7Szhanghch05         asyncResp->res.jsonValue["Members@odata.count"] = 1;
22761476687dSEd Tanous         nlohmann::json::array_t members;
22771476687dSEd Tanous         nlohmann::json& bmc = members.emplace_back();
22781476687dSEd Tanous         bmc["@odata.id"] = "/redfish/v1/Managers/bmc";
22791476687dSEd Tanous         asyncResp->res.jsonValue["Members"] = std::move(members);
22807e860f15SJohn Edward Broadbent         });
22819c310685SBorawski.Lukasz }
22829c310685SBorawski.Lukasz } // namespace redfish
2283