1 /*
2 // Copyright (c) 2018 Intel Corporation
3 //
4 // Licensed under the Apache License, Version 2.0 (the "License");
5 // you may not use this file except in compliance with the License.
6 // You may obtain a copy of the License at
7 //
8 //      http://www.apache.org/licenses/LICENSE-2.0
9 //
10 // Unless required by applicable law or agreed to in writing, software
11 // distributed under the License is distributed on an "AS IS" BASIS,
12 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 // See the License for the specific language governing permissions and
14 // limitations under the License.
15 */
16 #pragma once
17 
18 #include "bmcweb_config.h"
19 
20 #include "app.hpp"
21 #include "dbus_utility.hpp"
22 #include "health.hpp"
23 #include "query.hpp"
24 #include "redfish_util.hpp"
25 #include "registries/privilege_registry.hpp"
26 #include "utils/dbus_utils.hpp"
27 #include "utils/json_utils.hpp"
28 #include "utils/sw_utils.hpp"
29 #include "utils/systemd_utils.hpp"
30 #include "utils/time_utils.hpp"
31 
32 #include <boost/system/error_code.hpp>
33 #include <boost/url/format.hpp>
34 #include <sdbusplus/asio/property.hpp>
35 #include <sdbusplus/unpack_properties.hpp>
36 
37 #include <algorithm>
38 #include <array>
39 #include <cstdint>
40 #include <memory>
41 #include <optional>
42 #include <ranges>
43 #include <sstream>
44 #include <string>
45 #include <string_view>
46 #include <variant>
47 
48 namespace redfish
49 {
50 
51 /**
52  * Function reboots the BMC.
53  *
54  * @param[in] asyncResp - Shared pointer for completing asynchronous calls
55  */
56 inline void
57     doBMCGracefulRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
58 {
59     const char* processName = "xyz.openbmc_project.State.BMC";
60     const char* objectPath = "/xyz/openbmc_project/state/bmc0";
61     const char* interfaceName = "xyz.openbmc_project.State.BMC";
62     const std::string& propertyValue =
63         "xyz.openbmc_project.State.BMC.Transition.Reboot";
64     const char* destProperty = "RequestedBMCTransition";
65 
66     // Create the D-Bus variant for D-Bus call.
67     sdbusplus::asio::setProperty(
68         *crow::connections::systemBus, processName, objectPath, interfaceName,
69         destProperty, propertyValue,
70         [asyncResp](const boost::system::error_code& ec) {
71         // Use "Set" method to set the property value.
72         if (ec)
73         {
74             BMCWEB_LOG_DEBUG("[Set] Bad D-Bus request error: {}", ec);
75             messages::internalError(asyncResp->res);
76             return;
77         }
78 
79         messages::success(asyncResp->res);
80     });
81 }
82 
83 inline void
84     doBMCForceRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
85 {
86     const char* processName = "xyz.openbmc_project.State.BMC";
87     const char* objectPath = "/xyz/openbmc_project/state/bmc0";
88     const char* interfaceName = "xyz.openbmc_project.State.BMC";
89     const std::string& propertyValue =
90         "xyz.openbmc_project.State.BMC.Transition.HardReboot";
91     const char* destProperty = "RequestedBMCTransition";
92 
93     // Create the D-Bus variant for D-Bus call.
94     sdbusplus::asio::setProperty(
95         *crow::connections::systemBus, processName, objectPath, interfaceName,
96         destProperty, propertyValue,
97         [asyncResp](const boost::system::error_code& ec) {
98         // Use "Set" method to set the property value.
99         if (ec)
100         {
101             BMCWEB_LOG_DEBUG("[Set] Bad D-Bus request error: {}", ec);
102             messages::internalError(asyncResp->res);
103             return;
104         }
105 
106         messages::success(asyncResp->res);
107     });
108 }
109 
110 /**
111  * ManagerResetAction class supports the POST method for the Reset (reboot)
112  * action.
113  */
114 inline void requestRoutesManagerResetAction(App& app)
115 {
116     /**
117      * Function handles POST method request.
118      * Analyzes POST body before sending Reset (Reboot) request data to D-Bus.
119      * OpenBMC supports ResetType "GracefulRestart" and "ForceRestart".
120      */
121 
122     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/")
123         .privileges(redfish::privileges::postManager)
124         .methods(boost::beast::http::verb::post)(
125             [&app](const crow::Request& req,
126                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
127         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
128         {
129             return;
130         }
131         BMCWEB_LOG_DEBUG("Post Manager Reset.");
132 
133         std::string resetType;
134 
135         if (!json_util::readJsonAction(req, asyncResp->res, "ResetType",
136                                        resetType))
137         {
138             return;
139         }
140 
141         if (resetType == "GracefulRestart")
142         {
143             BMCWEB_LOG_DEBUG("Proceeding with {}", resetType);
144             doBMCGracefulRestart(asyncResp);
145             return;
146         }
147         if (resetType == "ForceRestart")
148         {
149             BMCWEB_LOG_DEBUG("Proceeding with {}", resetType);
150             doBMCForceRestart(asyncResp);
151             return;
152         }
153         BMCWEB_LOG_DEBUG("Invalid property value for ResetType: {}", resetType);
154         messages::actionParameterNotSupported(asyncResp->res, resetType,
155                                               "ResetType");
156 
157         return;
158     });
159 }
160 
161 /**
162  * ManagerResetToDefaultsAction class supports POST method for factory reset
163  * action.
164  */
165 inline void requestRoutesManagerResetToDefaultsAction(App& app)
166 {
167     /**
168      * Function handles ResetToDefaults POST method request.
169      *
170      * Analyzes POST body message and factory resets BMC by calling
171      * BMC code updater factory reset followed by a BMC reboot.
172      *
173      * BMC code updater factory reset wipes the whole BMC read-write
174      * filesystem which includes things like the network settings.
175      *
176      * OpenBMC only supports ResetToDefaultsType "ResetAll".
177      */
178 
179     BMCWEB_ROUTE(app,
180                  "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults/")
181         .privileges(redfish::privileges::postManager)
182         .methods(boost::beast::http::verb::post)(
183             [&app](const crow::Request& req,
184                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
185         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
186         {
187             return;
188         }
189         BMCWEB_LOG_DEBUG("Post ResetToDefaults.");
190 
191         std::optional<std::string> resetType;
192         std::optional<std::string> resetToDefaultsType;
193 
194         if (!json_util::readJsonAction(req, asyncResp->res, "ResetType",
195                                        resetType, "ResetToDefaultsType",
196                                        resetToDefaultsType))
197         {
198             BMCWEB_LOG_DEBUG("Missing property ResetType.");
199 
200             messages::actionParameterMissing(asyncResp->res, "ResetToDefaults",
201                                              "ResetType");
202             return;
203         }
204 
205         if (resetToDefaultsType && !resetType)
206         {
207             BMCWEB_LOG_WARNING(
208                 "Using deprecated ResetToDefaultsType, should be ResetType."
209                 "Support for the ResetToDefaultsType will be dropped in 2Q24");
210             resetType = resetToDefaultsType;
211         }
212 
213         if (resetType != "ResetAll")
214         {
215             BMCWEB_LOG_DEBUG("Invalid property value for ResetType: {}",
216                              *resetType);
217             messages::actionParameterNotSupported(asyncResp->res, *resetType,
218                                                   "ResetType");
219             return;
220         }
221 
222         crow::connections::systemBus->async_method_call(
223             [asyncResp](const boost::system::error_code& ec) {
224             if (ec)
225             {
226                 BMCWEB_LOG_DEBUG("Failed to ResetToDefaults: {}", ec);
227                 messages::internalError(asyncResp->res);
228                 return;
229             }
230             // Factory Reset doesn't actually happen until a reboot
231             // Can't erase what the BMC is running on
232             doBMCGracefulRestart(asyncResp);
233         },
234             "xyz.openbmc_project.Software.BMC.Updater",
235             "/xyz/openbmc_project/software",
236             "xyz.openbmc_project.Common.FactoryReset", "Reset");
237     });
238 }
239 
240 /**
241  * ManagerResetActionInfo derived class for delivering Manager
242  * ResetType AllowableValues using ResetInfo schema.
243  */
244 inline void requestRoutesManagerResetActionInfo(App& app)
245 {
246     /**
247      * Functions triggers appropriate requests on DBus
248      */
249 
250     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/ResetActionInfo/")
251         .privileges(redfish::privileges::getActionInfo)
252         .methods(boost::beast::http::verb::get)(
253             [&app](const crow::Request& req,
254                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
255         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
256         {
257             return;
258         }
259 
260         asyncResp->res.jsonValue["@odata.type"] =
261             "#ActionInfo.v1_1_2.ActionInfo";
262         asyncResp->res.jsonValue["@odata.id"] =
263             "/redfish/v1/Managers/bmc/ResetActionInfo";
264         asyncResp->res.jsonValue["Name"] = "Reset Action Info";
265         asyncResp->res.jsonValue["Id"] = "ResetActionInfo";
266         nlohmann::json::object_t parameter;
267         parameter["Name"] = "ResetType";
268         parameter["Required"] = true;
269         parameter["DataType"] = "String";
270 
271         nlohmann::json::array_t allowableValues;
272         allowableValues.emplace_back("GracefulRestart");
273         allowableValues.emplace_back("ForceRestart");
274         parameter["AllowableValues"] = std::move(allowableValues);
275 
276         nlohmann::json::array_t parameters;
277         parameters.emplace_back(std::move(parameter));
278 
279         asyncResp->res.jsonValue["Parameters"] = std::move(parameters);
280     });
281 }
282 
283 static constexpr const char* objectManagerIface =
284     "org.freedesktop.DBus.ObjectManager";
285 static constexpr const char* pidConfigurationIface =
286     "xyz.openbmc_project.Configuration.Pid";
287 static constexpr const char* pidZoneConfigurationIface =
288     "xyz.openbmc_project.Configuration.Pid.Zone";
289 static constexpr const char* stepwiseConfigurationIface =
290     "xyz.openbmc_project.Configuration.Stepwise";
291 static constexpr const char* thermalModeIface =
292     "xyz.openbmc_project.Control.ThermalMode";
293 
294 inline void
295     asyncPopulatePid(const std::string& connection, const std::string& path,
296                      const std::string& currentProfile,
297                      const std::vector<std::string>& supportedProfiles,
298                      const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
299 {
300     sdbusplus::message::object_path objPath(path);
301     dbus::utility::getManagedObjects(
302         connection, objPath,
303         [asyncResp, currentProfile, supportedProfiles](
304             const boost::system::error_code& ec,
305             const dbus::utility::ManagedObjectType& managedObj) {
306         if (ec)
307         {
308             BMCWEB_LOG_ERROR("{}", ec);
309             messages::internalError(asyncResp->res);
310             return;
311         }
312         nlohmann::json& configRoot =
313             asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"];
314         nlohmann::json& fans = configRoot["FanControllers"];
315         fans["@odata.type"] = "#OemManager.FanControllers";
316         fans["@odata.id"] =
317             "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanControllers";
318 
319         nlohmann::json& pids = configRoot["PidControllers"];
320         pids["@odata.type"] = "#OemManager.PidControllers";
321         pids["@odata.id"] =
322             "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers";
323 
324         nlohmann::json& stepwise = configRoot["StepwiseControllers"];
325         stepwise["@odata.type"] = "#OemManager.StepwiseControllers";
326         stepwise["@odata.id"] =
327             "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers";
328 
329         nlohmann::json& zones = configRoot["FanZones"];
330         zones["@odata.id"] =
331             "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones";
332         zones["@odata.type"] = "#OemManager.FanZones";
333         configRoot["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan";
334         configRoot["@odata.type"] = "#OemManager.Fan";
335         configRoot["Profile@Redfish.AllowableValues"] = supportedProfiles;
336 
337         if (!currentProfile.empty())
338         {
339             configRoot["Profile"] = currentProfile;
340         }
341         BMCWEB_LOG_DEBUG("profile = {} !", currentProfile);
342 
343         for (const auto& pathPair : managedObj)
344         {
345             for (const auto& intfPair : pathPair.second)
346             {
347                 if (intfPair.first != pidConfigurationIface &&
348                     intfPair.first != pidZoneConfigurationIface &&
349                     intfPair.first != stepwiseConfigurationIface)
350                 {
351                     continue;
352                 }
353 
354                 std::string name;
355 
356                 for (const std::pair<std::string,
357                                      dbus::utility::DbusVariantType>& propPair :
358                      intfPair.second)
359                 {
360                     if (propPair.first == "Name")
361                     {
362                         const std::string* namePtr =
363                             std::get_if<std::string>(&propPair.second);
364                         if (namePtr == nullptr)
365                         {
366                             BMCWEB_LOG_ERROR("Pid Name Field illegal");
367                             messages::internalError(asyncResp->res);
368                             return;
369                         }
370                         name = *namePtr;
371                         dbus::utility::escapePathForDbus(name);
372                     }
373                     else if (propPair.first == "Profiles")
374                     {
375                         const std::vector<std::string>* profiles =
376                             std::get_if<std::vector<std::string>>(
377                                 &propPair.second);
378                         if (profiles == nullptr)
379                         {
380                             BMCWEB_LOG_ERROR("Pid Profiles Field illegal");
381                             messages::internalError(asyncResp->res);
382                             return;
383                         }
384                         if (std::find(profiles->begin(), profiles->end(),
385                                       currentProfile) == profiles->end())
386                         {
387                             BMCWEB_LOG_INFO(
388                                 "{} not supported in current profile", name);
389                             continue;
390                         }
391                     }
392                 }
393                 nlohmann::json* config = nullptr;
394                 const std::string* classPtr = nullptr;
395 
396                 for (const std::pair<std::string,
397                                      dbus::utility::DbusVariantType>& propPair :
398                      intfPair.second)
399                 {
400                     if (propPair.first == "Class")
401                     {
402                         classPtr = std::get_if<std::string>(&propPair.second);
403                     }
404                 }
405 
406                 boost::urls::url url("/redfish/v1/Managers/bmc");
407                 if (intfPair.first == pidZoneConfigurationIface)
408                 {
409                     std::string chassis;
410                     if (!dbus::utility::getNthStringFromPath(pathPair.first.str,
411                                                              5, chassis))
412                     {
413                         chassis = "#IllegalValue";
414                     }
415                     nlohmann::json& zone = zones[name];
416                     zone["Chassis"]["@odata.id"] =
417                         boost::urls::format("/redfish/v1/Chassis/{}", chassis);
418                     url.set_fragment(
419                         ("/Oem/OpenBmc/Fan/FanZones"_json_pointer / name)
420                             .to_string());
421                     zone["@odata.id"] = std::move(url);
422                     zone["@odata.type"] = "#OemManager.FanZone";
423                     config = &zone;
424                 }
425 
426                 else if (intfPair.first == stepwiseConfigurationIface)
427                 {
428                     if (classPtr == nullptr)
429                     {
430                         BMCWEB_LOG_ERROR("Pid Class Field illegal");
431                         messages::internalError(asyncResp->res);
432                         return;
433                     }
434 
435                     nlohmann::json& controller = stepwise[name];
436                     config = &controller;
437                     url.set_fragment(
438                         ("/Oem/OpenBmc/Fan/StepwiseControllers"_json_pointer /
439                          name)
440                             .to_string());
441                     controller["@odata.id"] = std::move(url);
442                     controller["@odata.type"] =
443                         "#OemManager.StepwiseController";
444 
445                     controller["Direction"] = *classPtr;
446                 }
447 
448                 // pid and fans are off the same configuration
449                 else if (intfPair.first == pidConfigurationIface)
450                 {
451                     if (classPtr == nullptr)
452                     {
453                         BMCWEB_LOG_ERROR("Pid Class Field illegal");
454                         messages::internalError(asyncResp->res);
455                         return;
456                     }
457                     bool isFan = *classPtr == "fan";
458                     nlohmann::json& element = isFan ? fans[name] : pids[name];
459                     config = &element;
460                     if (isFan)
461                     {
462                         url.set_fragment(
463                             ("/Oem/OpenBmc/Fan/FanControllers"_json_pointer /
464                              name)
465                                 .to_string());
466                         element["@odata.id"] = std::move(url);
467                         element["@odata.type"] = "#OemManager.FanController";
468                     }
469                     else
470                     {
471                         url.set_fragment(
472                             ("/Oem/OpenBmc/Fan/PidControllers"_json_pointer /
473                              name)
474                                 .to_string());
475                         element["@odata.id"] = std::move(url);
476                         element["@odata.type"] = "#OemManager.PidController";
477                     }
478                 }
479                 else
480                 {
481                     BMCWEB_LOG_ERROR("Unexpected configuration");
482                     messages::internalError(asyncResp->res);
483                     return;
484                 }
485 
486                 // used for making maps out of 2 vectors
487                 const std::vector<double>* keys = nullptr;
488                 const std::vector<double>* values = nullptr;
489 
490                 for (const auto& propertyPair : intfPair.second)
491                 {
492                     if (propertyPair.first == "Type" ||
493                         propertyPair.first == "Class" ||
494                         propertyPair.first == "Name")
495                     {
496                         continue;
497                     }
498 
499                     // zones
500                     if (intfPair.first == pidZoneConfigurationIface)
501                     {
502                         const double* ptr =
503                             std::get_if<double>(&propertyPair.second);
504                         if (ptr == nullptr)
505                         {
506                             BMCWEB_LOG_ERROR("Field Illegal {}",
507                                              propertyPair.first);
508                             messages::internalError(asyncResp->res);
509                             return;
510                         }
511                         (*config)[propertyPair.first] = *ptr;
512                     }
513 
514                     if (intfPair.first == stepwiseConfigurationIface)
515                     {
516                         if (propertyPair.first == "Reading" ||
517                             propertyPair.first == "Output")
518                         {
519                             const std::vector<double>* ptr =
520                                 std::get_if<std::vector<double>>(
521                                     &propertyPair.second);
522 
523                             if (ptr == nullptr)
524                             {
525                                 BMCWEB_LOG_ERROR("Field Illegal {}",
526                                                  propertyPair.first);
527                                 messages::internalError(asyncResp->res);
528                                 return;
529                             }
530 
531                             if (propertyPair.first == "Reading")
532                             {
533                                 keys = ptr;
534                             }
535                             else
536                             {
537                                 values = ptr;
538                             }
539                             if (keys != nullptr && values != nullptr)
540                             {
541                                 if (keys->size() != values->size())
542                                 {
543                                     BMCWEB_LOG_ERROR(
544                                         "Reading and Output size don't match ");
545                                     messages::internalError(asyncResp->res);
546                                     return;
547                                 }
548                                 nlohmann::json& steps = (*config)["Steps"];
549                                 steps = nlohmann::json::array();
550                                 for (size_t ii = 0; ii < keys->size(); ii++)
551                                 {
552                                     nlohmann::json::object_t step;
553                                     step["Target"] = (*keys)[ii];
554                                     step["Output"] = (*values)[ii];
555                                     steps.emplace_back(std::move(step));
556                                 }
557                             }
558                         }
559                         if (propertyPair.first == "NegativeHysteresis" ||
560                             propertyPair.first == "PositiveHysteresis")
561                         {
562                             const double* ptr =
563                                 std::get_if<double>(&propertyPair.second);
564                             if (ptr == nullptr)
565                             {
566                                 BMCWEB_LOG_ERROR("Field Illegal {}",
567                                                  propertyPair.first);
568                                 messages::internalError(asyncResp->res);
569                                 return;
570                             }
571                             (*config)[propertyPair.first] = *ptr;
572                         }
573                     }
574 
575                     // pid and fans are off the same configuration
576                     if (intfPair.first == pidConfigurationIface ||
577                         intfPair.first == stepwiseConfigurationIface)
578                     {
579                         if (propertyPair.first == "Zones")
580                         {
581                             const std::vector<std::string>* inputs =
582                                 std::get_if<std::vector<std::string>>(
583                                     &propertyPair.second);
584 
585                             if (inputs == nullptr)
586                             {
587                                 BMCWEB_LOG_ERROR("Zones Pid Field Illegal");
588                                 messages::internalError(asyncResp->res);
589                                 return;
590                             }
591                             auto& data = (*config)[propertyPair.first];
592                             data = nlohmann::json::array();
593                             for (std::string itemCopy : *inputs)
594                             {
595                                 dbus::utility::escapePathForDbus(itemCopy);
596                                 nlohmann::json::object_t input;
597                                 boost::urls::url managerUrl = boost::urls::format(
598                                     "/redfish/v1/Managers/bmc#{}",
599                                     ("/Oem/OpenBmc/Fan/FanZones"_json_pointer /
600                                      itemCopy)
601                                         .to_string());
602                                 input["@odata.id"] = std::move(managerUrl);
603                                 data.emplace_back(std::move(input));
604                             }
605                         }
606                         // todo(james): may never happen, but this
607                         // assumes configuration data referenced in the
608                         // PID config is provided by the same daemon, we
609                         // could add another loop to cover all cases,
610                         // but I'm okay kicking this can down the road a
611                         // bit
612 
613                         else if (propertyPair.first == "Inputs" ||
614                                  propertyPair.first == "Outputs")
615                         {
616                             auto& data = (*config)[propertyPair.first];
617                             const std::vector<std::string>* inputs =
618                                 std::get_if<std::vector<std::string>>(
619                                     &propertyPair.second);
620 
621                             if (inputs == nullptr)
622                             {
623                                 BMCWEB_LOG_ERROR("Field Illegal {}",
624                                                  propertyPair.first);
625                                 messages::internalError(asyncResp->res);
626                                 return;
627                             }
628                             data = *inputs;
629                         }
630                         else if (propertyPair.first == "SetPointOffset")
631                         {
632                             const std::string* ptr =
633                                 std::get_if<std::string>(&propertyPair.second);
634 
635                             if (ptr == nullptr)
636                             {
637                                 BMCWEB_LOG_ERROR("Field Illegal {}",
638                                                  propertyPair.first);
639                                 messages::internalError(asyncResp->res);
640                                 return;
641                             }
642                             // translate from dbus to redfish
643                             if (*ptr == "WarningHigh")
644                             {
645                                 (*config)["SetPointOffset"] =
646                                     "UpperThresholdNonCritical";
647                             }
648                             else if (*ptr == "WarningLow")
649                             {
650                                 (*config)["SetPointOffset"] =
651                                     "LowerThresholdNonCritical";
652                             }
653                             else if (*ptr == "CriticalHigh")
654                             {
655                                 (*config)["SetPointOffset"] =
656                                     "UpperThresholdCritical";
657                             }
658                             else if (*ptr == "CriticalLow")
659                             {
660                                 (*config)["SetPointOffset"] =
661                                     "LowerThresholdCritical";
662                             }
663                             else
664                             {
665                                 BMCWEB_LOG_ERROR("Value Illegal {}", *ptr);
666                                 messages::internalError(asyncResp->res);
667                                 return;
668                             }
669                         }
670                         // doubles
671                         else if (propertyPair.first == "FFGainCoefficient" ||
672                                  propertyPair.first == "FFOffCoefficient" ||
673                                  propertyPair.first == "ICoefficient" ||
674                                  propertyPair.first == "ILimitMax" ||
675                                  propertyPair.first == "ILimitMin" ||
676                                  propertyPair.first == "PositiveHysteresis" ||
677                                  propertyPair.first == "NegativeHysteresis" ||
678                                  propertyPair.first == "OutLimitMax" ||
679                                  propertyPair.first == "OutLimitMin" ||
680                                  propertyPair.first == "PCoefficient" ||
681                                  propertyPair.first == "SetPoint" ||
682                                  propertyPair.first == "SlewNeg" ||
683                                  propertyPair.first == "SlewPos")
684                         {
685                             const double* ptr =
686                                 std::get_if<double>(&propertyPair.second);
687                             if (ptr == nullptr)
688                             {
689                                 BMCWEB_LOG_ERROR("Field Illegal {}",
690                                                  propertyPair.first);
691                                 messages::internalError(asyncResp->res);
692                                 return;
693                             }
694                             (*config)[propertyPair.first] = *ptr;
695                         }
696                     }
697                 }
698             }
699         }
700     });
701 }
702 
703 enum class CreatePIDRet
704 {
705     fail,
706     del,
707     patch
708 };
709 
710 inline bool
711     getZonesFromJsonReq(const std::shared_ptr<bmcweb::AsyncResp>& response,
712                         std::vector<nlohmann::json>& config,
713                         std::vector<std::string>& zones)
714 {
715     if (config.empty())
716     {
717         BMCWEB_LOG_ERROR("Empty Zones");
718         messages::propertyValueFormatError(response->res, config, "Zones");
719         return false;
720     }
721     for (auto& odata : config)
722     {
723         std::string path;
724         if (!redfish::json_util::readJson(odata, response->res, "@odata.id",
725                                           path))
726         {
727             return false;
728         }
729         std::string input;
730 
731         // 8 below comes from
732         // /redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/Left
733         //     0    1     2      3    4    5      6     7      8
734         if (!dbus::utility::getNthStringFromPath(path, 8, input))
735         {
736             BMCWEB_LOG_ERROR("Got invalid path {}", path);
737             BMCWEB_LOG_ERROR("Illegal Type Zones");
738             messages::propertyValueFormatError(response->res, odata, "Zones");
739             return false;
740         }
741         std::replace(input.begin(), input.end(), '_', ' ');
742         zones.emplace_back(std::move(input));
743     }
744     return true;
745 }
746 
747 inline const dbus::utility::ManagedObjectType::value_type*
748     findChassis(const dbus::utility::ManagedObjectType& managedObj,
749                 const std::string& value, std::string& chassis)
750 {
751     BMCWEB_LOG_DEBUG("Find Chassis: {}", value);
752 
753     std::string escaped = value;
754     std::replace(escaped.begin(), escaped.end(), ' ', '_');
755     escaped = "/" + escaped;
756     auto it = std::ranges::find_if(managedObj, [&escaped](const auto& obj) {
757         if (obj.first.str.ends_with(escaped))
758         {
759             BMCWEB_LOG_DEBUG("Matched {}", obj.first.str);
760             return true;
761         }
762         return false;
763     });
764 
765     if (it == managedObj.end())
766     {
767         return nullptr;
768     }
769     // 5 comes from <chassis-name> being the 5th element
770     // /xyz/openbmc_project/inventory/system/chassis/<chassis-name>
771     if (dbus::utility::getNthStringFromPath(it->first.str, 5, chassis))
772     {
773         return &(*it);
774     }
775 
776     return nullptr;
777 }
778 
779 inline CreatePIDRet createPidInterface(
780     const std::shared_ptr<bmcweb::AsyncResp>& response, const std::string& type,
781     const nlohmann::json::iterator& it, const std::string& path,
782     const dbus::utility::ManagedObjectType& managedObj, bool createNewObject,
783     dbus::utility::DBusPropertiesMap& output, std::string& chassis,
784     const std::string& profile)
785 {
786     // common deleter
787     if (it.value() == nullptr)
788     {
789         std::string iface;
790         if (type == "PidControllers" || type == "FanControllers")
791         {
792             iface = pidConfigurationIface;
793         }
794         else if (type == "FanZones")
795         {
796             iface = pidZoneConfigurationIface;
797         }
798         else if (type == "StepwiseControllers")
799         {
800             iface = stepwiseConfigurationIface;
801         }
802         else
803         {
804             BMCWEB_LOG_ERROR("Illegal Type {}", type);
805             messages::propertyUnknown(response->res, type);
806             return CreatePIDRet::fail;
807         }
808 
809         BMCWEB_LOG_DEBUG("del {} {}", path, iface);
810         // delete interface
811         crow::connections::systemBus->async_method_call(
812             [response, path](const boost::system::error_code& ec) {
813             if (ec)
814             {
815                 BMCWEB_LOG_ERROR("Error patching {}: {}", path, ec);
816                 messages::internalError(response->res);
817                 return;
818             }
819             messages::success(response->res);
820         },
821             "xyz.openbmc_project.EntityManager", path, iface, "Delete");
822         return CreatePIDRet::del;
823     }
824 
825     const dbus::utility::ManagedObjectType::value_type* managedItem = nullptr;
826     if (!createNewObject)
827     {
828         // if we aren't creating a new object, we should be able to find it on
829         // d-bus
830         managedItem = findChassis(managedObj, it.key(), chassis);
831         if (managedItem == nullptr)
832         {
833             BMCWEB_LOG_ERROR("Failed to get chassis from config patch");
834             messages::invalidObject(
835                 response->res,
836                 boost::urls::format("/redfish/v1/Chassis/{}", chassis));
837             return CreatePIDRet::fail;
838         }
839     }
840 
841     if (!profile.empty() &&
842         (type == "PidControllers" || type == "FanControllers" ||
843          type == "StepwiseControllers"))
844     {
845         if (managedItem == nullptr)
846         {
847             output.emplace_back("Profiles", std::vector<std::string>{profile});
848         }
849         else
850         {
851             std::string interface;
852             if (type == "StepwiseControllers")
853             {
854                 interface = stepwiseConfigurationIface;
855             }
856             else
857             {
858                 interface = pidConfigurationIface;
859             }
860             bool ifaceFound = false;
861             for (const auto& iface : managedItem->second)
862             {
863                 if (iface.first == interface)
864                 {
865                     ifaceFound = true;
866                     for (const auto& prop : iface.second)
867                     {
868                         if (prop.first == "Profiles")
869                         {
870                             const std::vector<std::string>* curProfiles =
871                                 std::get_if<std::vector<std::string>>(
872                                     &(prop.second));
873                             if (curProfiles == nullptr)
874                             {
875                                 BMCWEB_LOG_ERROR(
876                                     "Illegal profiles in managed object");
877                                 messages::internalError(response->res);
878                                 return CreatePIDRet::fail;
879                             }
880                             if (std::find(curProfiles->begin(),
881                                           curProfiles->end(),
882                                           profile) == curProfiles->end())
883                             {
884                                 std::vector<std::string> newProfiles =
885                                     *curProfiles;
886                                 newProfiles.push_back(profile);
887                                 output.emplace_back("Profiles", newProfiles);
888                             }
889                         }
890                     }
891                 }
892             }
893 
894             if (!ifaceFound)
895             {
896                 BMCWEB_LOG_ERROR("Failed to find interface in managed object");
897                 messages::internalError(response->res);
898                 return CreatePIDRet::fail;
899             }
900         }
901     }
902 
903     if (type == "PidControllers" || type == "FanControllers")
904     {
905         if (createNewObject)
906         {
907             output.emplace_back("Class",
908                                 type == "PidControllers" ? "temp" : "fan");
909             output.emplace_back("Type", "Pid");
910         }
911 
912         std::optional<std::vector<nlohmann::json>> zones;
913         std::optional<std::vector<std::string>> inputs;
914         std::optional<std::vector<std::string>> outputs;
915         std::map<std::string, std::optional<double>> doubles;
916         std::optional<std::string> setpointOffset;
917         if (!redfish::json_util::readJson(
918                 it.value(), response->res, "Inputs", inputs, "Outputs", outputs,
919                 "Zones", zones, "FFGainCoefficient",
920                 doubles["FFGainCoefficient"], "FFOffCoefficient",
921                 doubles["FFOffCoefficient"], "ICoefficient",
922                 doubles["ICoefficient"], "ILimitMax", doubles["ILimitMax"],
923                 "ILimitMin", doubles["ILimitMin"], "OutLimitMax",
924                 doubles["OutLimitMax"], "OutLimitMin", doubles["OutLimitMin"],
925                 "PCoefficient", doubles["PCoefficient"], "SetPoint",
926                 doubles["SetPoint"], "SetPointOffset", setpointOffset,
927                 "SlewNeg", doubles["SlewNeg"], "SlewPos", doubles["SlewPos"],
928                 "PositiveHysteresis", doubles["PositiveHysteresis"],
929                 "NegativeHysteresis", doubles["NegativeHysteresis"]))
930         {
931             return CreatePIDRet::fail;
932         }
933         if (zones)
934         {
935             std::vector<std::string> zonesStr;
936             if (!getZonesFromJsonReq(response, *zones, zonesStr))
937             {
938                 BMCWEB_LOG_ERROR("Illegal Zones");
939                 return CreatePIDRet::fail;
940             }
941             if (chassis.empty() &&
942                 findChassis(managedObj, zonesStr[0], chassis) == nullptr)
943             {
944                 BMCWEB_LOG_ERROR("Failed to get chassis from config patch");
945                 messages::invalidObject(
946                     response->res,
947                     boost::urls::format("/redfish/v1/Chassis/{}", chassis));
948                 return CreatePIDRet::fail;
949             }
950             output.emplace_back("Zones", std::move(zonesStr));
951         }
952 
953         if (inputs)
954         {
955             for (std::string& value : *inputs)
956             {
957                 std::replace(value.begin(), value.end(), '_', ' ');
958             }
959             output.emplace_back("Inputs", *inputs);
960         }
961 
962         if (outputs)
963         {
964             for (std::string& value : *outputs)
965             {
966                 std::replace(value.begin(), value.end(), '_', ' ');
967             }
968             output.emplace_back("Outputs", *outputs);
969         }
970 
971         if (setpointOffset)
972         {
973             // translate between redfish and dbus names
974             if (*setpointOffset == "UpperThresholdNonCritical")
975             {
976                 output.emplace_back("SetPointOffset", "WarningLow");
977             }
978             else if (*setpointOffset == "LowerThresholdNonCritical")
979             {
980                 output.emplace_back("SetPointOffset", "WarningHigh");
981             }
982             else if (*setpointOffset == "LowerThresholdCritical")
983             {
984                 output.emplace_back("SetPointOffset", "CriticalLow");
985             }
986             else if (*setpointOffset == "UpperThresholdCritical")
987             {
988                 output.emplace_back("SetPointOffset", "CriticalHigh");
989             }
990             else
991             {
992                 BMCWEB_LOG_ERROR("Invalid setpointoffset {}", *setpointOffset);
993                 messages::propertyValueNotInList(response->res, it.key(),
994                                                  "SetPointOffset");
995                 return CreatePIDRet::fail;
996             }
997         }
998 
999         // doubles
1000         for (const auto& pairs : doubles)
1001         {
1002             if (!pairs.second)
1003             {
1004                 continue;
1005             }
1006             BMCWEB_LOG_DEBUG("{} = {}", pairs.first, *pairs.second);
1007             output.emplace_back(pairs.first, *pairs.second);
1008         }
1009     }
1010 
1011     else if (type == "FanZones")
1012     {
1013         output.emplace_back("Type", "Pid.Zone");
1014 
1015         std::optional<nlohmann::json> chassisContainer;
1016         std::optional<double> failSafePercent;
1017         std::optional<double> minThermalOutput;
1018         if (!redfish::json_util::readJson(it.value(), response->res, "Chassis",
1019                                           chassisContainer, "FailSafePercent",
1020                                           failSafePercent, "MinThermalOutput",
1021                                           minThermalOutput))
1022         {
1023             return CreatePIDRet::fail;
1024         }
1025 
1026         if (chassisContainer)
1027         {
1028             std::string chassisId;
1029             if (!redfish::json_util::readJson(*chassisContainer, response->res,
1030                                               "@odata.id", chassisId))
1031             {
1032                 return CreatePIDRet::fail;
1033             }
1034 
1035             // /redfish/v1/chassis/chassis_name/
1036             if (!dbus::utility::getNthStringFromPath(chassisId, 3, chassis))
1037             {
1038                 BMCWEB_LOG_ERROR("Got invalid path {}", chassisId);
1039                 messages::invalidObject(
1040                     response->res,
1041                     boost::urls::format("/redfish/v1/Chassis/{}", chassisId));
1042                 return CreatePIDRet::fail;
1043             }
1044         }
1045         if (minThermalOutput)
1046         {
1047             output.emplace_back("MinThermalOutput", *minThermalOutput);
1048         }
1049         if (failSafePercent)
1050         {
1051             output.emplace_back("FailSafePercent", *failSafePercent);
1052         }
1053     }
1054     else if (type == "StepwiseControllers")
1055     {
1056         output.emplace_back("Type", "Stepwise");
1057 
1058         std::optional<std::vector<nlohmann::json>> zones;
1059         std::optional<std::vector<nlohmann::json>> steps;
1060         std::optional<std::vector<std::string>> inputs;
1061         std::optional<double> positiveHysteresis;
1062         std::optional<double> negativeHysteresis;
1063         std::optional<std::string> direction; // upper clipping curve vs lower
1064         if (!redfish::json_util::readJson(
1065                 it.value(), response->res, "Zones", zones, "Steps", steps,
1066                 "Inputs", inputs, "PositiveHysteresis", positiveHysteresis,
1067                 "NegativeHysteresis", negativeHysteresis, "Direction",
1068                 direction))
1069         {
1070             return CreatePIDRet::fail;
1071         }
1072 
1073         if (zones)
1074         {
1075             std::vector<std::string> zonesStrs;
1076             if (!getZonesFromJsonReq(response, *zones, zonesStrs))
1077             {
1078                 BMCWEB_LOG_ERROR("Illegal Zones");
1079                 return CreatePIDRet::fail;
1080             }
1081             if (chassis.empty() &&
1082                 findChassis(managedObj, zonesStrs[0], chassis) == nullptr)
1083             {
1084                 BMCWEB_LOG_ERROR("Failed to get chassis from config patch");
1085                 messages::invalidObject(
1086                     response->res,
1087                     boost::urls::format("/redfish/v1/Chassis/{}", chassis));
1088                 return CreatePIDRet::fail;
1089             }
1090             output.emplace_back("Zones", std::move(zonesStrs));
1091         }
1092         if (steps)
1093         {
1094             std::vector<double> readings;
1095             std::vector<double> outputs;
1096             for (auto& step : *steps)
1097             {
1098                 double target = 0.0;
1099                 double out = 0.0;
1100 
1101                 if (!redfish::json_util::readJson(step, response->res, "Target",
1102                                                   target, "Output", out))
1103                 {
1104                     return CreatePIDRet::fail;
1105                 }
1106                 readings.emplace_back(target);
1107                 outputs.emplace_back(out);
1108             }
1109             output.emplace_back("Reading", std::move(readings));
1110             output.emplace_back("Output", std::move(outputs));
1111         }
1112         if (inputs)
1113         {
1114             for (std::string& value : *inputs)
1115             {
1116                 std::replace(value.begin(), value.end(), '_', ' ');
1117             }
1118             output.emplace_back("Inputs", std::move(*inputs));
1119         }
1120         if (negativeHysteresis)
1121         {
1122             output.emplace_back("NegativeHysteresis", *negativeHysteresis);
1123         }
1124         if (positiveHysteresis)
1125         {
1126             output.emplace_back("PositiveHysteresis", *positiveHysteresis);
1127         }
1128         if (direction)
1129         {
1130             constexpr const std::array<const char*, 2> allowedDirections = {
1131                 "Ceiling", "Floor"};
1132             if (std::ranges::find(allowedDirections, *direction) ==
1133                 allowedDirections.end())
1134             {
1135                 messages::propertyValueTypeError(response->res, "Direction",
1136                                                  *direction);
1137                 return CreatePIDRet::fail;
1138             }
1139             output.emplace_back("Class", *direction);
1140         }
1141     }
1142     else
1143     {
1144         BMCWEB_LOG_ERROR("Illegal Type {}", type);
1145         messages::propertyUnknown(response->res, type);
1146         return CreatePIDRet::fail;
1147     }
1148     return CreatePIDRet::patch;
1149 }
1150 struct GetPIDValues : std::enable_shared_from_this<GetPIDValues>
1151 {
1152     struct CompletionValues
1153     {
1154         std::vector<std::string> supportedProfiles;
1155         std::string currentProfile;
1156         dbus::utility::MapperGetSubTreeResponse subtree;
1157     };
1158 
1159     explicit GetPIDValues(
1160         const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn) :
1161         asyncResp(asyncRespIn)
1162 
1163     {}
1164 
1165     void run()
1166     {
1167         std::shared_ptr<GetPIDValues> self = shared_from_this();
1168 
1169         // get all configurations
1170         constexpr std::array<std::string_view, 4> interfaces = {
1171             pidConfigurationIface, pidZoneConfigurationIface,
1172             objectManagerIface, stepwiseConfigurationIface};
1173         dbus::utility::getSubTree(
1174             "/", 0, interfaces,
1175             [self](
1176                 const boost::system::error_code& ec,
1177                 const dbus::utility::MapperGetSubTreeResponse& subtreeLocal) {
1178             if (ec)
1179             {
1180                 BMCWEB_LOG_ERROR("{}", ec);
1181                 messages::internalError(self->asyncResp->res);
1182                 return;
1183             }
1184             self->complete.subtree = subtreeLocal;
1185         });
1186 
1187         // at the same time get the selected profile
1188         constexpr std::array<std::string_view, 1> thermalModeIfaces = {
1189             thermalModeIface};
1190         dbus::utility::getSubTree(
1191             "/", 0, thermalModeIfaces,
1192             [self](
1193                 const boost::system::error_code& ec,
1194                 const dbus::utility::MapperGetSubTreeResponse& subtreeLocal) {
1195             if (ec || subtreeLocal.empty())
1196             {
1197                 return;
1198             }
1199             if (subtreeLocal[0].second.size() != 1)
1200             {
1201                 // invalid mapper response, should never happen
1202                 BMCWEB_LOG_ERROR("GetPIDValues: Mapper Error");
1203                 messages::internalError(self->asyncResp->res);
1204                 return;
1205             }
1206 
1207             const std::string& path = subtreeLocal[0].first;
1208             const std::string& owner = subtreeLocal[0].second[0].first;
1209 
1210             sdbusplus::asio::getAllProperties(
1211                 *crow::connections::systemBus, owner, path, thermalModeIface,
1212                 [path, owner,
1213                  self](const boost::system::error_code& ec2,
1214                        const dbus::utility::DBusPropertiesMap& resp) {
1215                 if (ec2)
1216                 {
1217                     BMCWEB_LOG_ERROR(
1218                         "GetPIDValues: Can't get thermalModeIface {}", path);
1219                     messages::internalError(self->asyncResp->res);
1220                     return;
1221                 }
1222 
1223                 const std::string* current = nullptr;
1224                 const std::vector<std::string>* supported = nullptr;
1225 
1226                 const bool success = sdbusplus::unpackPropertiesNoThrow(
1227                     dbus_utils::UnpackErrorPrinter(), resp, "Current", current,
1228                     "Supported", supported);
1229 
1230                 if (!success)
1231                 {
1232                     messages::internalError(self->asyncResp->res);
1233                     return;
1234                 }
1235 
1236                 if (current == nullptr || supported == nullptr)
1237                 {
1238                     BMCWEB_LOG_ERROR(
1239                         "GetPIDValues: thermal mode iface invalid {}", path);
1240                     messages::internalError(self->asyncResp->res);
1241                     return;
1242                 }
1243                 self->complete.currentProfile = *current;
1244                 self->complete.supportedProfiles = *supported;
1245             });
1246         });
1247     }
1248 
1249     static void
1250         processingComplete(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1251                            const CompletionValues& completion)
1252     {
1253         if (asyncResp->res.result() != boost::beast::http::status::ok)
1254         {
1255             return;
1256         }
1257         // create map of <connection, path to objMgr>>
1258         boost::container::flat_map<
1259             std::string, std::string, std::less<>,
1260             std::vector<std::pair<std::string, std::string>>>
1261             objectMgrPaths;
1262         boost::container::flat_set<std::string, std::less<>,
1263                                    std::vector<std::string>>
1264             calledConnections;
1265         for (const auto& pathGroup : completion.subtree)
1266         {
1267             for (const auto& connectionGroup : pathGroup.second)
1268             {
1269                 auto findConnection =
1270                     calledConnections.find(connectionGroup.first);
1271                 if (findConnection != calledConnections.end())
1272                 {
1273                     break;
1274                 }
1275                 for (const std::string& interface : connectionGroup.second)
1276                 {
1277                     if (interface == objectManagerIface)
1278                     {
1279                         objectMgrPaths[connectionGroup.first] = pathGroup.first;
1280                     }
1281                     // this list is alphabetical, so we
1282                     // should have found the objMgr by now
1283                     if (interface == pidConfigurationIface ||
1284                         interface == pidZoneConfigurationIface ||
1285                         interface == stepwiseConfigurationIface)
1286                     {
1287                         auto findObjMgr =
1288                             objectMgrPaths.find(connectionGroup.first);
1289                         if (findObjMgr == objectMgrPaths.end())
1290                         {
1291                             BMCWEB_LOG_DEBUG("{}Has no Object Manager",
1292                                              connectionGroup.first);
1293                             continue;
1294                         }
1295 
1296                         calledConnections.insert(connectionGroup.first);
1297 
1298                         asyncPopulatePid(findObjMgr->first, findObjMgr->second,
1299                                          completion.currentProfile,
1300                                          completion.supportedProfiles,
1301                                          asyncResp);
1302                         break;
1303                     }
1304                 }
1305             }
1306         }
1307     }
1308 
1309     ~GetPIDValues()
1310     {
1311         boost::asio::post(crow::connections::systemBus->get_io_context(),
1312                           std::bind_front(&processingComplete, asyncResp,
1313                                           std::move(complete)));
1314     }
1315 
1316     GetPIDValues(const GetPIDValues&) = delete;
1317     GetPIDValues(GetPIDValues&&) = delete;
1318     GetPIDValues& operator=(const GetPIDValues&) = delete;
1319     GetPIDValues& operator=(GetPIDValues&&) = delete;
1320 
1321     std::shared_ptr<bmcweb::AsyncResp> asyncResp;
1322     CompletionValues complete;
1323 };
1324 
1325 struct SetPIDValues : std::enable_shared_from_this<SetPIDValues>
1326 {
1327     SetPIDValues(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn,
1328                  nlohmann::json& data) :
1329         asyncResp(asyncRespIn)
1330     {
1331         std::optional<nlohmann::json> pidControllers;
1332         std::optional<nlohmann::json> fanControllers;
1333         std::optional<nlohmann::json> fanZones;
1334         std::optional<nlohmann::json> stepwiseControllers;
1335 
1336         if (!redfish::json_util::readJson(
1337                 data, asyncResp->res, "PidControllers", pidControllers,
1338                 "FanControllers", fanControllers, "FanZones", fanZones,
1339                 "StepwiseControllers", stepwiseControllers, "Profile", profile))
1340         {
1341             return;
1342         }
1343         configuration.emplace_back("PidControllers", std::move(pidControllers));
1344         configuration.emplace_back("FanControllers", std::move(fanControllers));
1345         configuration.emplace_back("FanZones", std::move(fanZones));
1346         configuration.emplace_back("StepwiseControllers",
1347                                    std::move(stepwiseControllers));
1348     }
1349 
1350     SetPIDValues(const SetPIDValues&) = delete;
1351     SetPIDValues(SetPIDValues&&) = delete;
1352     SetPIDValues& operator=(const SetPIDValues&) = delete;
1353     SetPIDValues& operator=(SetPIDValues&&) = delete;
1354 
1355     void run()
1356     {
1357         if (asyncResp->res.result() != boost::beast::http::status::ok)
1358         {
1359             return;
1360         }
1361 
1362         std::shared_ptr<SetPIDValues> self = shared_from_this();
1363 
1364         // todo(james): might make sense to do a mapper call here if this
1365         // interface gets more traction
1366         sdbusplus::message::object_path objPath(
1367             "/xyz/openbmc_project/inventory");
1368         dbus::utility::getManagedObjects(
1369             "xyz.openbmc_project.EntityManager", objPath,
1370             [self](const boost::system::error_code& ec,
1371                    const dbus::utility::ManagedObjectType& mObj) {
1372             if (ec)
1373             {
1374                 BMCWEB_LOG_ERROR("Error communicating to Entity Manager");
1375                 messages::internalError(self->asyncResp->res);
1376                 return;
1377             }
1378             const std::array<const char*, 3> configurations = {
1379                 pidConfigurationIface, pidZoneConfigurationIface,
1380                 stepwiseConfigurationIface};
1381 
1382             for (const auto& [path, object] : mObj)
1383             {
1384                 for (const auto& [interface, _] : object)
1385                 {
1386                     if (std::ranges::find(configurations, interface) !=
1387                         configurations.end())
1388                     {
1389                         self->objectCount++;
1390                         break;
1391                     }
1392                 }
1393             }
1394             self->managedObj = mObj;
1395         });
1396 
1397         // at the same time get the profile information
1398         constexpr std::array<std::string_view, 1> thermalModeIfaces = {
1399             thermalModeIface};
1400         dbus::utility::getSubTree(
1401             "/", 0, thermalModeIfaces,
1402             [self](const boost::system::error_code& ec,
1403                    const dbus::utility::MapperGetSubTreeResponse& subtree) {
1404             if (ec || subtree.empty())
1405             {
1406                 return;
1407             }
1408             if (subtree[0].second.empty())
1409             {
1410                 // invalid mapper response, should never happen
1411                 BMCWEB_LOG_ERROR("SetPIDValues: Mapper Error");
1412                 messages::internalError(self->asyncResp->res);
1413                 return;
1414             }
1415 
1416             const std::string& path = subtree[0].first;
1417             const std::string& owner = subtree[0].second[0].first;
1418             sdbusplus::asio::getAllProperties(
1419                 *crow::connections::systemBus, owner, path, thermalModeIface,
1420                 [self, path, owner](const boost::system::error_code& ec2,
1421                                     const dbus::utility::DBusPropertiesMap& r) {
1422                 if (ec2)
1423                 {
1424                     BMCWEB_LOG_ERROR(
1425                         "SetPIDValues: Can't get thermalModeIface {}", path);
1426                     messages::internalError(self->asyncResp->res);
1427                     return;
1428                 }
1429                 const std::string* current = nullptr;
1430                 const std::vector<std::string>* supported = nullptr;
1431 
1432                 const bool success = sdbusplus::unpackPropertiesNoThrow(
1433                     dbus_utils::UnpackErrorPrinter(), r, "Current", current,
1434                     "Supported", supported);
1435 
1436                 if (!success)
1437                 {
1438                     messages::internalError(self->asyncResp->res);
1439                     return;
1440                 }
1441 
1442                 if (current == nullptr || supported == nullptr)
1443                 {
1444                     BMCWEB_LOG_ERROR(
1445                         "SetPIDValues: thermal mode iface invalid {}", path);
1446                     messages::internalError(self->asyncResp->res);
1447                     return;
1448                 }
1449                 self->currentProfile = *current;
1450                 self->supportedProfiles = *supported;
1451                 self->profileConnection = owner;
1452                 self->profilePath = path;
1453             });
1454         });
1455     }
1456     void pidSetDone()
1457     {
1458         if (asyncResp->res.result() != boost::beast::http::status::ok)
1459         {
1460             return;
1461         }
1462         std::shared_ptr<bmcweb::AsyncResp> response = asyncResp;
1463         if (profile)
1464         {
1465             if (std::ranges::find(supportedProfiles, *profile) ==
1466                 supportedProfiles.end())
1467             {
1468                 messages::actionParameterUnknown(response->res, "Profile",
1469                                                  *profile);
1470                 return;
1471             }
1472             currentProfile = *profile;
1473             sdbusplus::asio::setProperty(
1474                 *crow::connections::systemBus, profileConnection, profilePath,
1475                 thermalModeIface, "Current", *profile,
1476                 [response](const boost::system::error_code& ec) {
1477                 if (ec)
1478                 {
1479                     BMCWEB_LOG_ERROR("Error patching profile{}", ec);
1480                     messages::internalError(response->res);
1481                 }
1482             });
1483         }
1484 
1485         for (auto& containerPair : configuration)
1486         {
1487             auto& container = containerPair.second;
1488             if (!container)
1489             {
1490                 continue;
1491             }
1492             BMCWEB_LOG_DEBUG("{}", *container);
1493 
1494             const std::string& type = containerPair.first;
1495 
1496             for (nlohmann::json::iterator it = container->begin();
1497                  it != container->end(); ++it)
1498             {
1499                 const auto& name = it.key();
1500                 std::string dbusObjName = name;
1501                 std::replace(dbusObjName.begin(), dbusObjName.end(), ' ', '_');
1502                 BMCWEB_LOG_DEBUG("looking for {}", name);
1503 
1504                 auto pathItr = std::ranges::find_if(
1505                     managedObj, [&dbusObjName](const auto& obj) {
1506                     return obj.first.parent_path() == dbusObjName;
1507                 });
1508                 dbus::utility::DBusPropertiesMap output;
1509 
1510                 output.reserve(16); // The pid interface length
1511 
1512                 // determines if we're patching entity-manager or
1513                 // creating a new object
1514                 bool createNewObject = (pathItr == managedObj.end());
1515                 BMCWEB_LOG_DEBUG("Found = {}", !createNewObject);
1516 
1517                 std::string iface;
1518                 if (!createNewObject)
1519                 {
1520                     bool findInterface = false;
1521                     for (const auto& interface : pathItr->second)
1522                     {
1523                         if (interface.first == pidConfigurationIface)
1524                         {
1525                             if (type == "PidControllers" ||
1526                                 type == "FanControllers")
1527                             {
1528                                 iface = pidConfigurationIface;
1529                                 findInterface = true;
1530                                 break;
1531                             }
1532                         }
1533                         else if (interface.first == pidZoneConfigurationIface)
1534                         {
1535                             if (type == "FanZones")
1536                             {
1537                                 iface = pidZoneConfigurationIface;
1538                                 findInterface = true;
1539                                 break;
1540                             }
1541                         }
1542                         else if (interface.first == stepwiseConfigurationIface)
1543                         {
1544                             if (type == "StepwiseControllers")
1545                             {
1546                                 iface = stepwiseConfigurationIface;
1547                                 findInterface = true;
1548                                 break;
1549                             }
1550                         }
1551                     }
1552 
1553                     // create new object if interface not found
1554                     if (!findInterface)
1555                     {
1556                         createNewObject = true;
1557                     }
1558                 }
1559 
1560                 if (createNewObject && it.value() == nullptr)
1561                 {
1562                     // can't delete a non-existent object
1563                     messages::propertyValueNotInList(response->res, it.value(),
1564                                                      name);
1565                     continue;
1566                 }
1567 
1568                 std::string path;
1569                 if (pathItr != managedObj.end())
1570                 {
1571                     path = pathItr->first.str;
1572                 }
1573 
1574                 BMCWEB_LOG_DEBUG("Create new = {}", createNewObject);
1575 
1576                 // arbitrary limit to avoid attacks
1577                 constexpr const size_t controllerLimit = 500;
1578                 if (createNewObject && objectCount >= controllerLimit)
1579                 {
1580                     messages::resourceExhaustion(response->res, type);
1581                     continue;
1582                 }
1583                 std::string escaped = name;
1584                 std::replace(escaped.begin(), escaped.end(), '_', ' ');
1585                 output.emplace_back("Name", escaped);
1586 
1587                 std::string chassis;
1588                 CreatePIDRet ret = createPidInterface(
1589                     response, type, it, path, managedObj, createNewObject,
1590                     output, chassis, currentProfile);
1591                 if (ret == CreatePIDRet::fail)
1592                 {
1593                     return;
1594                 }
1595                 if (ret == CreatePIDRet::del)
1596                 {
1597                     continue;
1598                 }
1599 
1600                 if (!createNewObject)
1601                 {
1602                     for (const auto& property : output)
1603                     {
1604                         crow::connections::systemBus->async_method_call(
1605                             [response,
1606                              propertyName{std::string(property.first)}](
1607                                 const boost::system::error_code& ec) {
1608                             if (ec)
1609                             {
1610                                 BMCWEB_LOG_ERROR("Error patching {}: {}",
1611                                                  propertyName, ec);
1612                                 messages::internalError(response->res);
1613                                 return;
1614                             }
1615                             messages::success(response->res);
1616                         },
1617                             "xyz.openbmc_project.EntityManager", path,
1618                             "org.freedesktop.DBus.Properties", "Set", iface,
1619                             property.first, property.second);
1620                     }
1621                 }
1622                 else
1623                 {
1624                     if (chassis.empty())
1625                     {
1626                         BMCWEB_LOG_ERROR("Failed to get chassis from config");
1627                         messages::internalError(response->res);
1628                         return;
1629                     }
1630 
1631                     bool foundChassis = false;
1632                     for (const auto& obj : managedObj)
1633                     {
1634                         if (obj.first.parent_path() == chassis)
1635                         {
1636                             chassis = obj.first.str;
1637                             foundChassis = true;
1638                             break;
1639                         }
1640                     }
1641                     if (!foundChassis)
1642                     {
1643                         BMCWEB_LOG_ERROR("Failed to find chassis on dbus");
1644                         messages::resourceMissingAtURI(
1645                             response->res,
1646                             boost::urls::format("/redfish/v1/Chassis/{}",
1647                                                 chassis));
1648                         return;
1649                     }
1650 
1651                     crow::connections::systemBus->async_method_call(
1652                         [response](const boost::system::error_code& ec) {
1653                         if (ec)
1654                         {
1655                             BMCWEB_LOG_ERROR("Error Adding Pid Object {}", ec);
1656                             messages::internalError(response->res);
1657                             return;
1658                         }
1659                         messages::success(response->res);
1660                     },
1661                         "xyz.openbmc_project.EntityManager", chassis,
1662                         "xyz.openbmc_project.AddObject", "AddObject", output);
1663                 }
1664             }
1665         }
1666     }
1667 
1668     ~SetPIDValues()
1669     {
1670         try
1671         {
1672             pidSetDone();
1673         }
1674         catch (...)
1675         {
1676             BMCWEB_LOG_CRITICAL("pidSetDone threw exception");
1677         }
1678     }
1679 
1680     std::shared_ptr<bmcweb::AsyncResp> asyncResp;
1681     std::vector<std::pair<std::string, std::optional<nlohmann::json>>>
1682         configuration;
1683     std::optional<std::string> profile;
1684     dbus::utility::ManagedObjectType managedObj;
1685     std::vector<std::string> supportedProfiles;
1686     std::string currentProfile;
1687     std::string profileConnection;
1688     std::string profilePath;
1689     size_t objectCount = 0;
1690 };
1691 
1692 /**
1693  * @brief Retrieves BMC manager location data over DBus
1694  *
1695  * @param[in] asyncResp Shared pointer for completing asynchronous calls
1696  * @param[in] connectionName - service name
1697  * @param[in] path - object path
1698  * @return none
1699  */
1700 inline void getLocation(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1701                         const std::string& connectionName,
1702                         const std::string& path)
1703 {
1704     BMCWEB_LOG_DEBUG("Get BMC manager Location data.");
1705 
1706     sdbusplus::asio::getProperty<std::string>(
1707         *crow::connections::systemBus, connectionName, path,
1708         "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode",
1709         [asyncResp](const boost::system::error_code& ec,
1710                     const std::string& property) {
1711         if (ec)
1712         {
1713             BMCWEB_LOG_DEBUG("DBUS response error for "
1714                              "Location");
1715             messages::internalError(asyncResp->res);
1716             return;
1717         }
1718 
1719         asyncResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] =
1720             property;
1721     });
1722 }
1723 // avoid name collision systems.hpp
1724 inline void
1725     managerGetLastResetTime(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
1726 {
1727     BMCWEB_LOG_DEBUG("Getting Manager Last Reset Time");
1728 
1729     sdbusplus::asio::getProperty<uint64_t>(
1730         *crow::connections::systemBus, "xyz.openbmc_project.State.BMC",
1731         "/xyz/openbmc_project/state/bmc0", "xyz.openbmc_project.State.BMC",
1732         "LastRebootTime",
1733         [asyncResp](const boost::system::error_code& ec,
1734                     const uint64_t lastResetTime) {
1735         if (ec)
1736         {
1737             BMCWEB_LOG_DEBUG("D-BUS response error {}", ec);
1738             return;
1739         }
1740 
1741         // LastRebootTime is epoch time, in milliseconds
1742         // https://github.com/openbmc/phosphor-dbus-interfaces/blob/7f9a128eb9296e926422ddc312c148b625890bb6/xyz/openbmc_project/State/BMC.interface.yaml#L19
1743         uint64_t lastResetTimeStamp = lastResetTime / 1000;
1744 
1745         // Convert to ISO 8601 standard
1746         asyncResp->res.jsonValue["LastResetTime"] =
1747             redfish::time_utils::getDateTimeUint(lastResetTimeStamp);
1748     });
1749 }
1750 
1751 /**
1752  * @brief Set the running firmware image
1753  *
1754  * @param[i,o] asyncResp - Async response object
1755  * @param[i] runningFirmwareTarget - Image to make the running image
1756  *
1757  * @return void
1758  */
1759 inline void
1760     setActiveFirmwareImage(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1761                            const std::string& runningFirmwareTarget)
1762 {
1763     // Get the Id from /redfish/v1/UpdateService/FirmwareInventory/<Id>
1764     std::string::size_type idPos = runningFirmwareTarget.rfind('/');
1765     if (idPos == std::string::npos)
1766     {
1767         messages::propertyValueNotInList(asyncResp->res, runningFirmwareTarget,
1768                                          "@odata.id");
1769         BMCWEB_LOG_DEBUG("Can't parse firmware ID!");
1770         return;
1771     }
1772     idPos++;
1773     if (idPos >= runningFirmwareTarget.size())
1774     {
1775         messages::propertyValueNotInList(asyncResp->res, runningFirmwareTarget,
1776                                          "@odata.id");
1777         BMCWEB_LOG_DEBUG("Invalid firmware ID.");
1778         return;
1779     }
1780     std::string firmwareId = runningFirmwareTarget.substr(idPos);
1781 
1782     // Make sure the image is valid before setting priority
1783     sdbusplus::message::object_path objPath("/xyz/openbmc_project/software");
1784     dbus::utility::getManagedObjects(
1785         "xyz.openbmc_project.Software.BMC.Updater", objPath,
1786         [asyncResp, firmwareId, runningFirmwareTarget](
1787             const boost::system::error_code& ec,
1788             const dbus::utility::ManagedObjectType& subtree) {
1789         if (ec)
1790         {
1791             BMCWEB_LOG_DEBUG("D-Bus response error getting objects.");
1792             messages::internalError(asyncResp->res);
1793             return;
1794         }
1795 
1796         if (subtree.empty())
1797         {
1798             BMCWEB_LOG_DEBUG("Can't find image!");
1799             messages::internalError(asyncResp->res);
1800             return;
1801         }
1802 
1803         bool foundImage = false;
1804         for (const auto& object : subtree)
1805         {
1806             const std::string& path =
1807                 static_cast<const std::string&>(object.first);
1808             std::size_t idPos2 = path.rfind('/');
1809 
1810             if (idPos2 == std::string::npos)
1811             {
1812                 continue;
1813             }
1814 
1815             idPos2++;
1816             if (idPos2 >= path.size())
1817             {
1818                 continue;
1819             }
1820 
1821             if (path.substr(idPos2) == firmwareId)
1822             {
1823                 foundImage = true;
1824                 break;
1825             }
1826         }
1827 
1828         if (!foundImage)
1829         {
1830             messages::propertyValueNotInList(
1831                 asyncResp->res, runningFirmwareTarget, "@odata.id");
1832             BMCWEB_LOG_DEBUG("Invalid firmware ID.");
1833             return;
1834         }
1835 
1836         BMCWEB_LOG_DEBUG("Setting firmware version {} to priority 0.",
1837                          firmwareId);
1838 
1839         // Only support Immediate
1840         // An addition could be a Redfish Setting like
1841         // ActiveSoftwareImageApplyTime and support OnReset
1842         sdbusplus::asio::setProperty(
1843             *crow::connections::systemBus,
1844             "xyz.openbmc_project.Software.BMC.Updater",
1845             "/xyz/openbmc_project/software/" + firmwareId,
1846             "xyz.openbmc_project.Software.RedundancyPriority", "Priority",
1847             static_cast<uint8_t>(0),
1848             [asyncResp](const boost::system::error_code& ec2) {
1849             if (ec2)
1850             {
1851                 BMCWEB_LOG_DEBUG("D-Bus response error setting.");
1852                 messages::internalError(asyncResp->res);
1853                 return;
1854             }
1855             doBMCGracefulRestart(asyncResp);
1856         });
1857     });
1858 }
1859 
1860 inline void
1861     afterSetDateTime(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1862                      const boost::system::error_code& ec,
1863                      const sdbusplus::message_t& msg)
1864 {
1865     if (ec)
1866     {
1867         BMCWEB_LOG_DEBUG("Failed to set elapsed time. DBUS response error {}",
1868                          ec);
1869         const sd_bus_error* dbusError = msg.get_error();
1870         if (dbusError != nullptr)
1871         {
1872             std::string_view errorName(dbusError->name);
1873             if (errorName ==
1874                 "org.freedesktop.timedate1.AutomaticTimeSyncEnabled")
1875             {
1876                 BMCWEB_LOG_DEBUG("Setting conflict");
1877                 messages::propertyValueConflict(
1878                     asyncResp->res, "DateTime",
1879                     "Managers/NetworkProtocol/NTPProcotolEnabled");
1880                 return;
1881             }
1882         }
1883         messages::internalError(asyncResp->res);
1884         return;
1885     }
1886     asyncResp->res.result(boost::beast::http::status::no_content);
1887 }
1888 
1889 inline void setDateTime(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1890                         const std::string& datetime)
1891 {
1892     BMCWEB_LOG_DEBUG("Set date time: {}", datetime);
1893 
1894     std::optional<redfish::time_utils::usSinceEpoch> us =
1895         redfish::time_utils::dateStringToEpoch(datetime);
1896     if (!us)
1897     {
1898         messages::propertyValueFormatError(asyncResp->res, datetime,
1899                                            "DateTime");
1900         return;
1901     }
1902     // Set the absolute datetime
1903     bool relative = false;
1904     bool interactive = false;
1905     crow::connections::systemBus->async_method_call(
1906         [asyncResp](const boost::system::error_code& ec,
1907                     const sdbusplus::message_t& msg) {
1908         afterSetDateTime(asyncResp, ec, msg);
1909     },
1910         "org.freedesktop.timedate1", "/org/freedesktop/timedate1",
1911         "org.freedesktop.timedate1", "SetTime", us->count(), relative,
1912         interactive);
1913 }
1914 
1915 inline void
1916     checkForQuiesced(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
1917 {
1918     sdbusplus::asio::getProperty<std::string>(
1919         *crow::connections::systemBus, "org.freedesktop.systemd1",
1920         "/org/freedesktop/systemd1/unit/obmc-bmc-service-quiesce@0.target",
1921         "org.freedesktop.systemd1.Unit", "ActiveState",
1922         [asyncResp](const boost::system::error_code& ec,
1923                     const std::string& val) {
1924         if (!ec)
1925         {
1926             if (val == "active")
1927             {
1928                 asyncResp->res.jsonValue["Status"]["Health"] = "Critical";
1929                 asyncResp->res.jsonValue["Status"]["State"] = "Quiesced";
1930                 return;
1931             }
1932         }
1933         asyncResp->res.jsonValue["Status"]["Health"] = "OK";
1934         asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
1935     });
1936 }
1937 
1938 inline void requestRoutesManager(App& app)
1939 {
1940     std::string uuid = persistent_data::getConfig().systemUuid;
1941 
1942     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/")
1943         .privileges(redfish::privileges::getManager)
1944         .methods(boost::beast::http::verb::get)(
1945             [&app, uuid](const crow::Request& req,
1946                          const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
1947         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
1948         {
1949             return;
1950         }
1951         asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc";
1952         asyncResp->res.jsonValue["@odata.type"] = "#Manager.v1_14_0.Manager";
1953         asyncResp->res.jsonValue["Id"] = "bmc";
1954         asyncResp->res.jsonValue["Name"] = "OpenBmc Manager";
1955         asyncResp->res.jsonValue["Description"] =
1956             "Baseboard Management Controller";
1957         asyncResp->res.jsonValue["PowerState"] = "On";
1958 
1959         asyncResp->res.jsonValue["ManagerType"] = "BMC";
1960         asyncResp->res.jsonValue["UUID"] = systemd_utils::getUuid();
1961         asyncResp->res.jsonValue["ServiceEntryPointUUID"] = uuid;
1962         asyncResp->res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model
1963 
1964         asyncResp->res.jsonValue["LogServices"]["@odata.id"] =
1965             "/redfish/v1/Managers/bmc/LogServices";
1966         asyncResp->res.jsonValue["NetworkProtocol"]["@odata.id"] =
1967             "/redfish/v1/Managers/bmc/NetworkProtocol";
1968         asyncResp->res.jsonValue["EthernetInterfaces"]["@odata.id"] =
1969             "/redfish/v1/Managers/bmc/EthernetInterfaces";
1970 
1971 #ifdef BMCWEB_ENABLE_VM_NBDPROXY
1972         asyncResp->res.jsonValue["VirtualMedia"]["@odata.id"] =
1973             "/redfish/v1/Managers/bmc/VirtualMedia";
1974 #endif // BMCWEB_ENABLE_VM_NBDPROXY
1975 
1976         // default oem data
1977         nlohmann::json& oem = asyncResp->res.jsonValue["Oem"];
1978         nlohmann::json& oemOpenbmc = oem["OpenBmc"];
1979         oem["@odata.type"] = "#OemManager.Oem";
1980         oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem";
1981         oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc";
1982         oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc";
1983 
1984         nlohmann::json::object_t certificates;
1985         certificates["@odata.id"] =
1986             "/redfish/v1/Managers/bmc/Truststore/Certificates";
1987         oemOpenbmc["Certificates"] = std::move(certificates);
1988 
1989         // Manager.Reset (an action) can be many values, OpenBMC only
1990         // supports BMC reboot.
1991         nlohmann::json& managerReset =
1992             asyncResp->res.jsonValue["Actions"]["#Manager.Reset"];
1993         managerReset["target"] =
1994             "/redfish/v1/Managers/bmc/Actions/Manager.Reset";
1995         managerReset["@Redfish.ActionInfo"] =
1996             "/redfish/v1/Managers/bmc/ResetActionInfo";
1997 
1998         // ResetToDefaults (Factory Reset) has values like
1999         // PreserveNetworkAndUsers and PreserveNetwork that aren't supported
2000         // on OpenBMC
2001         nlohmann::json& resetToDefaults =
2002             asyncResp->res.jsonValue["Actions"]["#Manager.ResetToDefaults"];
2003         resetToDefaults["target"] =
2004             "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults";
2005         resetToDefaults["ResetType@Redfish.AllowableValues"] =
2006             nlohmann::json::array_t({"ResetAll"});
2007 
2008         std::pair<std::string, std::string> redfishDateTimeOffset =
2009             redfish::time_utils::getDateTimeOffsetNow();
2010 
2011         asyncResp->res.jsonValue["DateTime"] = redfishDateTimeOffset.first;
2012         asyncResp->res.jsonValue["DateTimeLocalOffset"] =
2013             redfishDateTimeOffset.second;
2014 
2015         // TODO (Gunnar): Remove these one day since moved to ComputerSystem
2016         // Still used by OCP profiles
2017         // https://github.com/opencomputeproject/OCP-Profiles/issues/23
2018         // Fill in SerialConsole info
2019         asyncResp->res.jsonValue["SerialConsole"]["ServiceEnabled"] = true;
2020         asyncResp->res.jsonValue["SerialConsole"]["MaxConcurrentSessions"] = 15;
2021         asyncResp->res.jsonValue["SerialConsole"]["ConnectTypesSupported"] =
2022             nlohmann::json::array_t({"IPMI", "SSH"});
2023 #ifdef BMCWEB_ENABLE_KVM
2024         // Fill in GraphicalConsole info
2025         asyncResp->res.jsonValue["GraphicalConsole"]["ServiceEnabled"] = true;
2026         asyncResp->res.jsonValue["GraphicalConsole"]["MaxConcurrentSessions"] =
2027             4;
2028         asyncResp->res.jsonValue["GraphicalConsole"]["ConnectTypesSupported"] =
2029             nlohmann::json::array_t({"KVMIP"});
2030 #endif // BMCWEB_ENABLE_KVM
2031         if constexpr (!bmcwebEnableMultiHost)
2032         {
2033             asyncResp->res.jsonValue["Links"]["ManagerForServers@odata.count"] =
2034                 1;
2035 
2036             nlohmann::json::array_t managerForServers;
2037             nlohmann::json::object_t manager;
2038             manager["@odata.id"] = "/redfish/v1/Systems/system";
2039             managerForServers.emplace_back(std::move(manager));
2040 
2041             asyncResp->res.jsonValue["Links"]["ManagerForServers"] =
2042                 std::move(managerForServers);
2043         }
2044         if constexpr (bmcwebEnableHealthPopulate)
2045         {
2046             auto health = std::make_shared<HealthPopulate>(asyncResp);
2047             health->isManagersHealth = true;
2048             health->populate();
2049         }
2050 
2051         sw_util::populateSoftwareInformation(asyncResp, sw_util::bmcPurpose,
2052                                              "FirmwareVersion", true);
2053 
2054         managerGetLastResetTime(asyncResp);
2055 
2056         // ManagerDiagnosticData is added for all BMCs.
2057         nlohmann::json& managerDiagnosticData =
2058             asyncResp->res.jsonValue["ManagerDiagnosticData"];
2059         managerDiagnosticData["@odata.id"] =
2060             "/redfish/v1/Managers/bmc/ManagerDiagnosticData";
2061 
2062 #ifdef BMCWEB_ENABLE_REDFISH_OEM_MANAGER_FAN_DATA
2063         auto pids = std::make_shared<GetPIDValues>(asyncResp);
2064         pids->run();
2065 #endif
2066 
2067         getMainChassisId(asyncResp,
2068                          [](const std::string& chassisId,
2069                             const std::shared_ptr<bmcweb::AsyncResp>& aRsp) {
2070             aRsp->res.jsonValue["Links"]["ManagerForChassis@odata.count"] = 1;
2071             nlohmann::json::array_t managerForChassis;
2072             nlohmann::json::object_t managerObj;
2073             boost::urls::url chassiUrl =
2074                 boost::urls::format("/redfish/v1/Chassis/{}", chassisId);
2075             managerObj["@odata.id"] = chassiUrl;
2076             managerForChassis.emplace_back(std::move(managerObj));
2077             aRsp->res.jsonValue["Links"]["ManagerForChassis"] =
2078                 std::move(managerForChassis);
2079             aRsp->res.jsonValue["Links"]["ManagerInChassis"]["@odata.id"] =
2080                 chassiUrl;
2081         });
2082 
2083         sdbusplus::asio::getProperty<double>(
2084             *crow::connections::systemBus, "org.freedesktop.systemd1",
2085             "/org/freedesktop/systemd1", "org.freedesktop.systemd1.Manager",
2086             "Progress",
2087             [asyncResp](const boost::system::error_code& ec, double val) {
2088             if (ec)
2089             {
2090                 BMCWEB_LOG_ERROR("Error while getting progress");
2091                 messages::internalError(asyncResp->res);
2092                 return;
2093             }
2094             if (val < 1.0)
2095             {
2096                 asyncResp->res.jsonValue["Status"]["Health"] = "OK";
2097                 asyncResp->res.jsonValue["Status"]["State"] = "Starting";
2098                 return;
2099             }
2100             checkForQuiesced(asyncResp);
2101         });
2102 
2103         constexpr std::array<std::string_view, 1> interfaces = {
2104             "xyz.openbmc_project.Inventory.Item.Bmc"};
2105         dbus::utility::getSubTree(
2106             "/xyz/openbmc_project/inventory", 0, interfaces,
2107             [asyncResp](
2108                 const boost::system::error_code& ec,
2109                 const dbus::utility::MapperGetSubTreeResponse& subtree) {
2110             if (ec)
2111             {
2112                 BMCWEB_LOG_DEBUG("D-Bus response error on GetSubTree {}", ec);
2113                 return;
2114             }
2115             if (subtree.empty())
2116             {
2117                 BMCWEB_LOG_DEBUG("Can't find bmc D-Bus object!");
2118                 return;
2119             }
2120             // Assume only 1 bmc D-Bus object
2121             // Throw an error if there is more than 1
2122             if (subtree.size() > 1)
2123             {
2124                 BMCWEB_LOG_DEBUG("Found more than 1 bmc D-Bus object!");
2125                 messages::internalError(asyncResp->res);
2126                 return;
2127             }
2128 
2129             if (subtree[0].first.empty() || subtree[0].second.size() != 1)
2130             {
2131                 BMCWEB_LOG_DEBUG("Error getting bmc D-Bus object!");
2132                 messages::internalError(asyncResp->res);
2133                 return;
2134             }
2135 
2136             const std::string& path = subtree[0].first;
2137             const std::string& connectionName = subtree[0].second[0].first;
2138 
2139             for (const auto& interfaceName : subtree[0].second[0].second)
2140             {
2141                 if (interfaceName ==
2142                     "xyz.openbmc_project.Inventory.Decorator.Asset")
2143                 {
2144                     sdbusplus::asio::getAllProperties(
2145                         *crow::connections::systemBus, connectionName, path,
2146                         "xyz.openbmc_project.Inventory.Decorator.Asset",
2147                         [asyncResp](const boost::system::error_code& ec2,
2148                                     const dbus::utility::DBusPropertiesMap&
2149                                         propertiesList) {
2150                         if (ec2)
2151                         {
2152                             BMCWEB_LOG_DEBUG("Can't get bmc asset!");
2153                             return;
2154                         }
2155 
2156                         const std::string* partNumber = nullptr;
2157                         const std::string* serialNumber = nullptr;
2158                         const std::string* manufacturer = nullptr;
2159                         const std::string* model = nullptr;
2160                         const std::string* sparePartNumber = nullptr;
2161 
2162                         const bool success = sdbusplus::unpackPropertiesNoThrow(
2163                             dbus_utils::UnpackErrorPrinter(), propertiesList,
2164                             "PartNumber", partNumber, "SerialNumber",
2165                             serialNumber, "Manufacturer", manufacturer, "Model",
2166                             model, "SparePartNumber", sparePartNumber);
2167 
2168                         if (!success)
2169                         {
2170                             messages::internalError(asyncResp->res);
2171                             return;
2172                         }
2173 
2174                         if (partNumber != nullptr)
2175                         {
2176                             asyncResp->res.jsonValue["PartNumber"] =
2177                                 *partNumber;
2178                         }
2179 
2180                         if (serialNumber != nullptr)
2181                         {
2182                             asyncResp->res.jsonValue["SerialNumber"] =
2183                                 *serialNumber;
2184                         }
2185 
2186                         if (manufacturer != nullptr)
2187                         {
2188                             asyncResp->res.jsonValue["Manufacturer"] =
2189                                 *manufacturer;
2190                         }
2191 
2192                         if (model != nullptr)
2193                         {
2194                             asyncResp->res.jsonValue["Model"] = *model;
2195                         }
2196 
2197                         if (sparePartNumber != nullptr)
2198                         {
2199                             asyncResp->res.jsonValue["SparePartNumber"] =
2200                                 *sparePartNumber;
2201                         }
2202                     });
2203                 }
2204                 else if (interfaceName ==
2205                          "xyz.openbmc_project.Inventory.Decorator.LocationCode")
2206                 {
2207                     getLocation(asyncResp, connectionName, path);
2208                 }
2209             }
2210         });
2211     });
2212 
2213     BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/")
2214         .privileges(redfish::privileges::patchManager)
2215         .methods(boost::beast::http::verb::patch)(
2216             [&app](const crow::Request& req,
2217                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2218         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
2219         {
2220             return;
2221         }
2222         std::optional<nlohmann::json> oem;
2223         std::optional<nlohmann::json> links;
2224         std::optional<std::string> datetime;
2225 
2226         if (!json_util::readJsonPatch(req, asyncResp->res, "Oem", oem,
2227                                       "DateTime", datetime, "Links", links))
2228         {
2229             return;
2230         }
2231 
2232         if (oem)
2233         {
2234 #ifdef BMCWEB_ENABLE_REDFISH_OEM_MANAGER_FAN_DATA
2235             std::optional<nlohmann::json> openbmc;
2236             if (!redfish::json_util::readJson(*oem, asyncResp->res, "OpenBmc",
2237                                               openbmc))
2238             {
2239                 return;
2240             }
2241             if (openbmc)
2242             {
2243                 std::optional<nlohmann::json> fan;
2244                 if (!redfish::json_util::readJson(*openbmc, asyncResp->res,
2245                                                   "Fan", fan))
2246                 {
2247                     return;
2248                 }
2249                 if (fan)
2250                 {
2251                     auto pid = std::make_shared<SetPIDValues>(asyncResp, *fan);
2252                     pid->run();
2253                 }
2254             }
2255 #else
2256             messages::propertyUnknown(asyncResp->res, "Oem");
2257             return;
2258 #endif
2259         }
2260         if (links)
2261         {
2262             std::optional<nlohmann::json> activeSoftwareImage;
2263             if (!redfish::json_util::readJson(*links, asyncResp->res,
2264                                               "ActiveSoftwareImage",
2265                                               activeSoftwareImage))
2266             {
2267                 return;
2268             }
2269             if (activeSoftwareImage)
2270             {
2271                 std::optional<std::string> odataId;
2272                 if (!json_util::readJson(*activeSoftwareImage, asyncResp->res,
2273                                          "@odata.id", odataId))
2274                 {
2275                     return;
2276                 }
2277 
2278                 if (odataId)
2279                 {
2280                     setActiveFirmwareImage(asyncResp, *odataId);
2281                 }
2282             }
2283         }
2284         if (datetime)
2285         {
2286             setDateTime(asyncResp, *datetime);
2287         }
2288     });
2289 }
2290 
2291 inline void requestRoutesManagerCollection(App& app)
2292 {
2293     BMCWEB_ROUTE(app, "/redfish/v1/Managers/")
2294         .privileges(redfish::privileges::getManagerCollection)
2295         .methods(boost::beast::http::verb::get)(
2296             [&app](const crow::Request& req,
2297                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
2298         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
2299         {
2300             return;
2301         }
2302         // Collections don't include the static data added by SubRoute
2303         // because it has a duplicate entry for members
2304         asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers";
2305         asyncResp->res.jsonValue["@odata.type"] =
2306             "#ManagerCollection.ManagerCollection";
2307         asyncResp->res.jsonValue["Name"] = "Manager Collection";
2308         asyncResp->res.jsonValue["Members@odata.count"] = 1;
2309         nlohmann::json::array_t members;
2310         nlohmann::json& bmc = members.emplace_back();
2311         bmc["@odata.id"] = "/redfish/v1/Managers/bmc";
2312         asyncResp->res.jsonValue["Members"] = std::move(members);
2313     });
2314 }
2315 } // namespace redfish
2316