xref: /openbmc/bmcweb/redfish-core/lib/hypervisor_system.hpp (revision 8b24275d7696c9df071d11e16a0f905b1e800163)
1 #pragma once
2 
3 #include "app.hpp"
4 #include "dbus_singleton.hpp"
5 #include "dbus_utility.hpp"
6 #include "error_messages.hpp"
7 #include "ethernet.hpp"
8 #include "query.hpp"
9 #include "registries/privilege_registry.hpp"
10 #include "utils/ip_utils.hpp"
11 #include "utils/json_utils.hpp"
12 
13 #include <boost/url/format.hpp>
14 #include <sdbusplus/asio/property.hpp>
15 
16 #include <array>
17 #include <optional>
18 #include <string_view>
19 #include <utility>
20 
21 namespace redfish
22 {
23 
24 /**
25  * @brief Retrieves hypervisor state properties over dbus
26  *
27  * The hypervisor state object is optional so this function will only set the
28  * state variables if the object is found
29  *
30  * @param[in] asyncResp     Shared pointer for completing asynchronous calls.
31  *
32  * @return None.
33  */
34 inline void
35     getHypervisorState(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
36 {
37     BMCWEB_LOG_DEBUG << "Get hypervisor state information.";
38     sdbusplus::asio::getProperty<std::string>(
39         *crow::connections::systemBus, "xyz.openbmc_project.State.Hypervisor",
40         "/xyz/openbmc_project/state/hypervisor0",
41         "xyz.openbmc_project.State.Host", "CurrentHostState",
42         [asyncResp](const boost::system::error_code& ec,
43                     const std::string& hostState) {
44         if (ec)
45         {
46             BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
47             // This is an optional D-Bus object so just return if
48             // error occurs
49             return;
50         }
51 
52         BMCWEB_LOG_DEBUG << "Hypervisor state: " << hostState;
53         // Verify Host State
54         if (hostState == "xyz.openbmc_project.State.Host.HostState.Running")
55         {
56             asyncResp->res.jsonValue["PowerState"] = "On";
57             asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
58         }
59         else if (hostState == "xyz.openbmc_project.State.Host.HostState."
60                               "Quiesced")
61         {
62             asyncResp->res.jsonValue["PowerState"] = "On";
63             asyncResp->res.jsonValue["Status"]["State"] = "Quiesced";
64         }
65         else if (hostState == "xyz.openbmc_project.State.Host.HostState."
66                               "Standby")
67         {
68             asyncResp->res.jsonValue["PowerState"] = "On";
69             asyncResp->res.jsonValue["Status"]["State"] = "StandbyOffline";
70         }
71         else if (hostState == "xyz.openbmc_project.State.Host.HostState."
72                               "TransitioningToRunning")
73         {
74             asyncResp->res.jsonValue["PowerState"] = "PoweringOn";
75             asyncResp->res.jsonValue["Status"]["State"] = "Starting";
76         }
77         else if (hostState == "xyz.openbmc_project.State.Host.HostState."
78                               "TransitioningToOff")
79         {
80             asyncResp->res.jsonValue["PowerState"] = "PoweringOff";
81             asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
82         }
83         else if (hostState == "xyz.openbmc_project.State.Host.HostState.Off")
84         {
85             asyncResp->res.jsonValue["PowerState"] = "Off";
86             asyncResp->res.jsonValue["Status"]["State"] = "Disabled";
87         }
88         else
89         {
90             messages::internalError(asyncResp->res);
91             return;
92         }
93         });
94 }
95 
96 /**
97  * @brief Populate Actions if any are valid for hypervisor object
98  *
99  * The hypervisor state object is optional so this function will only set the
100  * Action if the object is found
101  *
102  * @param[in] asyncResp     Shared pointer for completing asynchronous calls.
103  *
104  * @return None.
105  */
106 inline void
107     getHypervisorActions(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
108 {
109     BMCWEB_LOG_DEBUG << "Get hypervisor actions.";
110     constexpr std::array<std::string_view, 1> interfaces = {
111         "xyz.openbmc_project.State.Host"};
112     dbus::utility::getDbusObject(
113         "/xyz/openbmc_project/state/hypervisor0", interfaces,
114         [asyncResp](
115             const boost::system::error_code& ec,
116             const std::vector<std::pair<std::string, std::vector<std::string>>>&
117                 objInfo) {
118         if (ec)
119         {
120             BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
121             // This is an optional D-Bus object so just return if
122             // error occurs
123             return;
124         }
125 
126         if (objInfo.empty())
127         {
128             // As noted above, this is an optional interface so just return
129             // if there is no instance found
130             return;
131         }
132 
133         if (objInfo.size() > 1)
134         {
135             // More then one hypervisor object is not supported and is an
136             // error
137             messages::internalError(asyncResp->res);
138             return;
139         }
140 
141         // Object present so system support limited ComputerSystem Action
142         nlohmann::json& reset =
143             asyncResp->res.jsonValue["Actions"]["#ComputerSystem.Reset"];
144         reset["target"] =
145             "/redfish/v1/Systems/hypervisor/Actions/ComputerSystem.Reset";
146         reset["@Redfish.ActionInfo"] =
147             "/redfish/v1/Systems/hypervisor/ResetActionInfo";
148         });
149 }
150 
151 inline bool extractHypervisorInterfaceData(
152     const std::string& ethIfaceId,
153     const dbus::utility::ManagedObjectType& dbusData,
154     EthernetInterfaceData& ethData, std::vector<IPv4AddressData>& ipv4Config)
155 {
156     bool idFound = false;
157     for (const auto& objpath : dbusData)
158     {
159         for (const auto& ifacePair : objpath.second)
160         {
161             if (objpath.first ==
162                 "/xyz/openbmc_project/network/hypervisor/" + ethIfaceId)
163             {
164                 idFound = true;
165                 if (ifacePair.first == "xyz.openbmc_project.Network.MACAddress")
166                 {
167                     for (const auto& propertyPair : ifacePair.second)
168                     {
169                         if (propertyPair.first == "MACAddress")
170                         {
171                             const std::string* mac =
172                                 std::get_if<std::string>(&propertyPair.second);
173                             if (mac != nullptr)
174                             {
175                                 ethData.macAddress = *mac;
176                             }
177                         }
178                     }
179                 }
180                 else if (ifacePair.first ==
181                          "xyz.openbmc_project.Network.EthernetInterface")
182                 {
183                     for (const auto& propertyPair : ifacePair.second)
184                     {
185                         if (propertyPair.first == "DHCPEnabled")
186                         {
187                             const std::string* dhcp =
188                                 std::get_if<std::string>(&propertyPair.second);
189                             if (dhcp != nullptr)
190                             {
191                                 ethData.dhcpEnabled = *dhcp;
192                                 break; // Interested on only "DHCPEnabled".
193                                        // Stop parsing since we got the
194                                        // "DHCPEnabled" value.
195                             }
196                         }
197                     }
198                 }
199             }
200             if (objpath.first == "/xyz/openbmc_project/network/hypervisor/" +
201                                      ethIfaceId + "/ipv4/addr0")
202             {
203                 IPv4AddressData& ipv4Address = ipv4Config.emplace_back();
204                 if (ifacePair.first == "xyz.openbmc_project.Object.Enable")
205                 {
206                     for (const auto& property : ifacePair.second)
207                     {
208                         if (property.first == "Enabled")
209                         {
210                             const bool* intfEnable =
211                                 std::get_if<bool>(&property.second);
212                             if (intfEnable != nullptr)
213                             {
214                                 ipv4Address.isActive = *intfEnable;
215                                 break;
216                             }
217                         }
218                     }
219                 }
220                 if (ifacePair.first == "xyz.openbmc_project.Network.IP")
221                 {
222                     for (const auto& property : ifacePair.second)
223                     {
224                         if (property.first == "Address")
225                         {
226                             const std::string* address =
227                                 std::get_if<std::string>(&property.second);
228                             if (address != nullptr)
229                             {
230                                 ipv4Address.address = *address;
231                             }
232                         }
233                         else if (property.first == "Origin")
234                         {
235                             const std::string* origin =
236                                 std::get_if<std::string>(&property.second);
237                             if (origin != nullptr)
238                             {
239                                 ipv4Address.origin =
240                                     translateAddressOriginDbusToRedfish(*origin,
241                                                                         true);
242                             }
243                         }
244                         else if (property.first == "PrefixLength")
245                         {
246                             const uint8_t* mask =
247                                 std::get_if<uint8_t>(&property.second);
248                             if (mask != nullptr)
249                             {
250                                 // convert it to the string
251                                 ipv4Address.netmask = getNetmask(*mask);
252                             }
253                         }
254                         else if (property.first == "Type" ||
255                                  property.first == "Gateway")
256                         {
257                             // Type & Gateway is not used
258                             continue;
259                         }
260                         else
261                         {
262                             BMCWEB_LOG_ERROR
263                                 << "Got extra property: " << property.first
264                                 << " on the " << objpath.first.str << " object";
265                         }
266                     }
267                 }
268             }
269             if (objpath.first == "/xyz/openbmc_project/network/hypervisor")
270             {
271                 // System configuration shows up in the global namespace, so no
272                 // need to check eth number
273                 if (ifacePair.first ==
274                     "xyz.openbmc_project.Network.SystemConfiguration")
275                 {
276                     for (const auto& propertyPair : ifacePair.second)
277                     {
278                         if (propertyPair.first == "HostName")
279                         {
280                             const std::string* hostName =
281                                 std::get_if<std::string>(&propertyPair.second);
282                             if (hostName != nullptr)
283                             {
284                                 ethData.hostName = *hostName;
285                             }
286                         }
287                         else if (propertyPair.first == "DefaultGateway")
288                         {
289                             const std::string* defaultGateway =
290                                 std::get_if<std::string>(&propertyPair.second);
291                             if (defaultGateway != nullptr)
292                             {
293                                 ethData.defaultGateway = *defaultGateway;
294                             }
295                         }
296                     }
297                 }
298             }
299         }
300     }
301     return idFound;
302 }
303 /**
304  * Function that retrieves all properties for given Hypervisor Ethernet
305  * Interface Object from Settings Manager
306  * @param ethIfaceId Hypervisor ethernet interface id to query on DBus
307  * @param callback a function that shall be called to convert Dbus output
308  * into JSON
309  */
310 template <typename CallbackFunc>
311 void getHypervisorIfaceData(const std::string& ethIfaceId,
312                             CallbackFunc&& callback)
313 {
314     sdbusplus::message::object_path path("/");
315     dbus::utility::getManagedObjects(
316         "xyz.openbmc_project.Settings", path,
317         [ethIfaceId{std::string{ethIfaceId}},
318          callback{std::forward<CallbackFunc>(callback)}](
319             const boost::system::error_code& ec,
320             const dbus::utility::ManagedObjectType& resp) {
321         EthernetInterfaceData ethData{};
322         std::vector<IPv4AddressData> ipv4Data;
323         if (ec)
324         {
325             callback(false, ethData, ipv4Data);
326             return;
327         }
328 
329         bool found = extractHypervisorInterfaceData(ethIfaceId, resp, ethData,
330                                                     ipv4Data);
331         if (!found)
332         {
333             BMCWEB_LOG_INFO << "Hypervisor Interface not found";
334         }
335         callback(found, ethData, ipv4Data);
336         });
337 }
338 
339 /**
340  * @brief Sets the Hypervisor Interface IPAddress DBUS
341  *
342  * @param[in] asyncResp          Shared pointer for generating response message.
343  * @param[in] ipv4Address    Address from the incoming request
344  * @param[in] ethIfaceId     Hypervisor Interface Id
345  *
346  * @return None.
347  */
348 inline void setHypervisorIPv4Address(
349     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
350     const std::string& ethIfaceId, const std::string& ipv4Address)
351 {
352     BMCWEB_LOG_DEBUG << "Setting the Hypervisor IPaddress : " << ipv4Address
353                      << " on Iface: " << ethIfaceId;
354     sdbusplus::asio::setProperty(
355         *crow::connections::systemBus, "xyz.openbmc_project.Settings",
356         "/xyz/openbmc_project/network/hypervisor/" + ethIfaceId + "/ipv4/addr0",
357         "xyz.openbmc_project.Network.IP", "Address", ipv4Address,
358         [asyncResp](const boost::system::error_code& ec) {
359         if (ec)
360         {
361             BMCWEB_LOG_ERROR << "DBUS response error " << ec;
362             return;
363         }
364         BMCWEB_LOG_DEBUG << "Hypervisor IPaddress is Set";
365         });
366 }
367 
368 /**
369  * @brief Sets the Hypervisor Interface SubnetMask DBUS
370  *
371  * @param[in] asyncResp     Shared pointer for generating response message.
372  * @param[in] subnet    SubnetMask from the incoming request
373  * @param[in] ethIfaceId Hypervisor Interface Id
374  *
375  * @return None.
376  */
377 inline void
378     setHypervisorIPv4Subnet(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
379                             const std::string& ethIfaceId, const uint8_t subnet)
380 {
381     BMCWEB_LOG_DEBUG << "Setting the Hypervisor subnet : " << subnet
382                      << " on Iface: " << ethIfaceId;
383 
384     sdbusplus::asio::setProperty(
385         *crow::connections::systemBus, "xyz.openbmc_project.Settings",
386         "/xyz/openbmc_project/network/hypervisor/" + ethIfaceId + "/ipv4/addr0",
387         "xyz.openbmc_project.Network.IP", "PrefixLength", subnet,
388         [asyncResp](const boost::system::error_code& ec) {
389         if (ec)
390         {
391             BMCWEB_LOG_ERROR << "DBUS response error " << ec;
392             return;
393         }
394         BMCWEB_LOG_DEBUG << "SubnetMask is Set";
395         });
396 }
397 
398 /**
399  * @brief Sets the Hypervisor Interface Gateway DBUS
400  *
401  * @param[in] asyncResp          Shared pointer for generating response message.
402  * @param[in] gateway        Gateway from the incoming request
403  * @param[in] ethIfaceId     Hypervisor Interface Id
404  *
405  * @return None.
406  */
407 inline void setHypervisorIPv4Gateway(
408     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
409     const std::string& gateway)
410 {
411     BMCWEB_LOG_DEBUG
412         << "Setting the DefaultGateway to the last configured gateway";
413 
414     sdbusplus::asio::setProperty(
415         *crow::connections::systemBus, "xyz.openbmc_project.Settings",
416         "/xyz/openbmc_project/network/hypervisor",
417         "xyz.openbmc_project.Network.SystemConfiguration", "DefaultGateway",
418         gateway, [asyncResp](const boost::system::error_code& ec) {
419             if (ec)
420             {
421                 BMCWEB_LOG_ERROR << "DBUS response error " << ec;
422                 return;
423             }
424             BMCWEB_LOG_DEBUG << "Default Gateway is Set";
425         });
426 }
427 
428 /**
429  * @brief Creates a static IPv4 entry
430  *
431  * @param[in] ifaceId      Id of interface upon which to create the IPv4 entry
432  * @param[in] prefixLength IPv4 prefix syntax for the subnet mask
433  * @param[in] gateway      IPv4 address of this interfaces gateway
434  * @param[in] address      IPv4 address to assign to this interface
435  * @param[io] asyncResp    Response object that will be returned to client
436  *
437  * @return None
438  */
439 inline void
440     createHypervisorIPv4(const std::string& ifaceId, uint8_t prefixLength,
441                          const std::string& gateway, const std::string& address,
442                          const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
443 {
444     setHypervisorIPv4Address(asyncResp, ifaceId, address);
445     setHypervisorIPv4Gateway(asyncResp, gateway);
446     setHypervisorIPv4Subnet(asyncResp, ifaceId, prefixLength);
447 }
448 
449 /**
450  * @brief Deletes given IPv4 interface
451  *
452  * @param[in] ifaceId     Id of interface whose IP should be deleted
453  * @param[io] asyncResp   Response object that will be returned to client
454  *
455  * @return None
456  */
457 inline void
458     deleteHypervisorIPv4(const std::string& ifaceId,
459                          const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
460 {
461     std::string address = "0.0.0.0";
462     std::string gateway = "0.0.0.0";
463     const uint8_t prefixLength = 0;
464     setHypervisorIPv4Address(asyncResp, ifaceId, address);
465     setHypervisorIPv4Gateway(asyncResp, gateway);
466     setHypervisorIPv4Subnet(asyncResp, ifaceId, prefixLength);
467 }
468 
469 inline void parseInterfaceData(nlohmann::json& jsonResponse,
470                                const std::string& ifaceId,
471                                const EthernetInterfaceData& ethData,
472                                const std::vector<IPv4AddressData>& ipv4Data)
473 {
474     jsonResponse["Id"] = ifaceId;
475     jsonResponse["@odata.id"] = boost::urls::format(
476         "/redfish/v1/Systems/hypervisor/EthernetInterfaces/{}", ifaceId);
477     jsonResponse["InterfaceEnabled"] = true;
478     jsonResponse["MACAddress"] = ethData.macAddress;
479 
480     jsonResponse["HostName"] = ethData.hostName;
481     jsonResponse["DHCPv4"]["DHCPEnabled"] =
482         translateDhcpEnabledToBool(ethData.dhcpEnabled, true);
483 
484     nlohmann::json& ipv4Array = jsonResponse["IPv4Addresses"];
485     nlohmann::json& ipv4StaticArray = jsonResponse["IPv4StaticAddresses"];
486     ipv4Array = nlohmann::json::array();
487     ipv4StaticArray = nlohmann::json::array();
488     for (const auto& ipv4Config : ipv4Data)
489     {
490         if (ipv4Config.isActive)
491         {
492             nlohmann::json::object_t ipv4;
493             ipv4["AddressOrigin"] = ipv4Config.origin;
494             ipv4["SubnetMask"] = ipv4Config.netmask;
495             ipv4["Address"] = ipv4Config.address;
496             ipv4["Gateway"] = ethData.defaultGateway;
497 
498             if (ipv4Config.origin == "Static")
499             {
500                 ipv4StaticArray.push_back(ipv4);
501             }
502             ipv4Array.emplace_back(std::move(ipv4));
503         }
504     }
505 }
506 
507 inline void setDHCPEnabled(const std::string& ifaceId, bool ipv4DHCPEnabled,
508                            const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
509 {
510     const std::string dhcp = getDhcpEnabledEnumeration(ipv4DHCPEnabled, false);
511     sdbusplus::asio::setProperty(
512         *crow::connections::systemBus, "xyz.openbmc_project.Settings",
513         "/xyz/openbmc_project/network/hypervisor/" + ifaceId,
514         "xyz.openbmc_project.Network.EthernetInterface", "DHCPEnabled", dhcp,
515         [asyncResp](const boost::system::error_code& ec) {
516         if (ec)
517         {
518             BMCWEB_LOG_ERROR << "D-Bus responses error: " << ec;
519             messages::internalError(asyncResp->res);
520             return;
521         }
522         });
523 
524     // Set the IPv4 address origin to the DHCP / Static as per the new value
525     // of the DHCPEnabled property
526     std::string origin;
527     if (!ipv4DHCPEnabled)
528     {
529         origin = "xyz.openbmc_project.Network.IP.AddressOrigin.Static";
530     }
531     else
532     {
533         // DHCPEnabled is set to true. Delete the current IPv4 settings
534         // to receive the new values from DHCP server.
535         deleteHypervisorIPv4(ifaceId, asyncResp);
536         origin = "xyz.openbmc_project.Network.IP.AddressOrigin.DHCP";
537     }
538     sdbusplus::asio::setProperty(
539         *crow::connections::systemBus, "xyz.openbmc_project.Settings",
540         "/xyz/openbmc_project/network/hypervisor/" + ifaceId + "/ipv4/addr0",
541         "xyz.openbmc_project.Network.IP", "Origin", origin,
542         [asyncResp](const boost::system::error_code& ec) {
543         if (ec)
544         {
545             BMCWEB_LOG_ERROR << "DBUS response error " << ec;
546             messages::internalError(asyncResp->res);
547             return;
548         }
549         BMCWEB_LOG_DEBUG << "Hypervisor IPaddress Origin is Set";
550         });
551 }
552 
553 inline void handleHypervisorIPv4StaticPatch(
554     const std::string& ifaceId, const nlohmann::json& input,
555     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
556 {
557     if ((!input.is_array()) || input.empty())
558     {
559         messages::propertyValueTypeError(asyncResp->res, input,
560                                          "IPv4StaticAddresses");
561         return;
562     }
563 
564     // Hypervisor considers the first IP address in the array list
565     // as the Hypervisor's virtual management interface supports single IPv4
566     // address
567     const nlohmann::json& thisJson = input[0];
568 
569     if (!thisJson.is_null() && !thisJson.empty())
570     {
571         // For the error string
572         std::string pathString = "IPv4StaticAddresses/1";
573         std::optional<std::string> address;
574         std::optional<std::string> subnetMask;
575         std::optional<std::string> gateway;
576         nlohmann::json thisJsonCopy = thisJson;
577         if (!json_util::readJson(thisJsonCopy, asyncResp->res, "Address",
578                                  address, "SubnetMask", subnetMask, "Gateway",
579                                  gateway))
580         {
581             messages::propertyValueFormatError(asyncResp->res, thisJson,
582                                                pathString);
583             return;
584         }
585 
586         uint8_t prefixLength = 0;
587         bool errorInEntry = false;
588         if (address)
589         {
590             if (!ip_util::ipv4VerifyIpAndGetBitcount(*address))
591             {
592                 messages::propertyValueFormatError(asyncResp->res, *address,
593                                                    pathString + "/Address");
594                 errorInEntry = true;
595             }
596         }
597         else
598         {
599             messages::propertyMissing(asyncResp->res, pathString + "/Address");
600             errorInEntry = true;
601         }
602 
603         if (subnetMask)
604         {
605             if (!ip_util::ipv4VerifyIpAndGetBitcount(*subnetMask,
606                                                      &prefixLength))
607             {
608                 messages::propertyValueFormatError(asyncResp->res, *subnetMask,
609                                                    pathString + "/SubnetMask");
610                 errorInEntry = true;
611             }
612         }
613         else
614         {
615             messages::propertyMissing(asyncResp->res,
616                                       pathString + "/SubnetMask");
617             errorInEntry = true;
618         }
619 
620         if (gateway)
621         {
622             if (!ip_util::ipv4VerifyIpAndGetBitcount(*gateway))
623             {
624                 messages::propertyValueFormatError(asyncResp->res, *gateway,
625                                                    pathString + "/Gateway");
626                 errorInEntry = true;
627             }
628         }
629         else
630         {
631             messages::propertyMissing(asyncResp->res, pathString + "/Gateway");
632             errorInEntry = true;
633         }
634 
635         if (errorInEntry)
636         {
637             return;
638         }
639 
640         BMCWEB_LOG_DEBUG << "Calling createHypervisorIPv4 on : " << ifaceId
641                          << "," << *address;
642         createHypervisorIPv4(ifaceId, prefixLength, *gateway, *address,
643                              asyncResp);
644         // Set the DHCPEnabled to false since the Static IPv4 is set
645         setDHCPEnabled(ifaceId, false, asyncResp);
646     }
647     else
648     {
649         if (thisJson.is_null())
650         {
651             deleteHypervisorIPv4(ifaceId, asyncResp);
652         }
653     }
654 }
655 
656 inline void handleHypervisorHostnamePatch(
657     const std::string& hostName,
658     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
659 {
660     if (!isHostnameValid(hostName))
661     {
662         messages::propertyValueFormatError(asyncResp->res, hostName,
663                                            "HostName");
664         return;
665     }
666 
667     asyncResp->res.jsonValue["HostName"] = hostName;
668     sdbusplus::asio::setProperty(
669         *crow::connections::systemBus, "xyz.openbmc_project.Settings",
670         "/xyz/openbmc_project/network/hypervisor/",
671         "xyz.openbmc_project.Network.SystemConfiguration", "HostName", hostName,
672         [asyncResp](const boost::system::error_code& ec) {
673         if (ec)
674         {
675             messages::internalError(asyncResp->res);
676         }
677         });
678 }
679 
680 inline void
681     setIPv4InterfaceEnabled(const std::string& ifaceId, bool isActive,
682                             const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
683 {
684     sdbusplus::asio::setProperty(
685         *crow::connections::systemBus, "xyz.openbmc_project.Settings",
686         "/xyz/openbmc_project/network/hypervisor/" + ifaceId + "/ipv4/addr0",
687         "xyz.openbmc_project.Object.Enable", "Enabled", isActive,
688         [asyncResp](const boost::system::error_code& ec) {
689         if (ec)
690         {
691             messages::internalError(asyncResp->res);
692         }
693         });
694 }
695 
696 inline void handleHypervisorEthernetInterfaceCollectionGet(
697     App& app, const crow::Request& req,
698     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
699 {
700     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
701     {
702         return;
703     }
704     constexpr std::array<std::string_view, 1> interfaces = {
705         "xyz.openbmc_project.Network.EthernetInterface"};
706 
707     dbus::utility::getSubTreePaths(
708         "/xyz/openbmc_project/network/hypervisor", 0, interfaces,
709         [asyncResp](
710             const boost::system::error_code& ec,
711             const dbus::utility::MapperGetSubTreePathsResponse& ifaceList) {
712         if (ec)
713         {
714             messages::resourceNotFound(asyncResp->res, "System", "hypervisor");
715             return;
716         }
717         asyncResp->res.jsonValue["@odata.type"] =
718             "#EthernetInterfaceCollection."
719             "EthernetInterfaceCollection";
720         asyncResp->res.jsonValue["@odata.id"] =
721             "/redfish/v1/Systems/hypervisor/EthernetInterfaces";
722         asyncResp->res.jsonValue["Name"] = "Hypervisor Ethernet "
723                                            "Interface Collection";
724         asyncResp->res.jsonValue["Description"] =
725             "Collection of Virtual Management "
726             "Interfaces for the hypervisor";
727 
728         nlohmann::json& ifaceArray = asyncResp->res.jsonValue["Members"];
729         ifaceArray = nlohmann::json::array();
730         for (const std::string& iface : ifaceList)
731         {
732             sdbusplus::message::object_path path(iface);
733             std::string name = path.filename();
734             if (name.empty())
735             {
736                 continue;
737             }
738             nlohmann::json::object_t ethIface;
739             ethIface["@odata.id"] = boost::urls::format(
740                 "/redfish/v1/Systems/hypervisor/EthernetInterfaces/{}", name);
741             ifaceArray.emplace_back(std::move(ethIface));
742         }
743         asyncResp->res.jsonValue["Members@odata.count"] = ifaceArray.size();
744         });
745 }
746 
747 inline void handleHypervisorEthernetInterfaceGet(
748     App& app, const crow::Request& req,
749     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, const std::string& id)
750 {
751     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
752     {
753         return;
754     }
755     getHypervisorIfaceData(
756         id, [asyncResp, ifaceId{std::string(id)}](
757                 bool success, const EthernetInterfaceData& ethData,
758                 const std::vector<IPv4AddressData>& ipv4Data) {
759             if (!success)
760             {
761                 messages::resourceNotFound(asyncResp->res, "EthernetInterface",
762                                            ifaceId);
763                 return;
764             }
765             asyncResp->res.jsonValue["@odata.type"] =
766                 "#EthernetInterface.v1_9_0.EthernetInterface";
767             asyncResp->res.jsonValue["Name"] = "Hypervisor Ethernet Interface";
768             asyncResp->res.jsonValue["Description"] =
769                 "Hypervisor's Virtual Management Ethernet Interface";
770             parseInterfaceData(asyncResp->res.jsonValue, ifaceId, ethData,
771                                ipv4Data);
772         });
773 }
774 
775 inline void handleHypervisorSystemGet(
776     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
777 {
778     sdbusplus::asio::getProperty<std::string>(
779         *crow::connections::systemBus, "xyz.openbmc_project.Settings",
780         "/xyz/openbmc_project/network/hypervisor",
781         "xyz.openbmc_project.Network.SystemConfiguration", "HostName",
782         [asyncResp](const boost::system::error_code& ec,
783                     const std::string& /*hostName*/) {
784         if (ec)
785         {
786             messages::resourceNotFound(asyncResp->res, "System", "hypervisor");
787             return;
788         }
789         BMCWEB_LOG_DEBUG << "Hypervisor is available";
790 
791         asyncResp->res.jsonValue["@odata.type"] =
792             "#ComputerSystem.v1_6_0.ComputerSystem";
793         asyncResp->res.jsonValue["@odata.id"] =
794             "/redfish/v1/Systems/hypervisor";
795         asyncResp->res.jsonValue["Description"] = "Hypervisor";
796         asyncResp->res.jsonValue["Name"] = "Hypervisor";
797         asyncResp->res.jsonValue["Id"] = "hypervisor";
798         asyncResp->res.jsonValue["SystemType"] = "OS";
799         nlohmann::json::array_t managedBy;
800         nlohmann::json::object_t manager;
801         manager["@odata.id"] = "/redfish/v1/Managers/bmc";
802         managedBy.emplace_back(std::move(manager));
803         asyncResp->res.jsonValue["Links"]["ManagedBy"] = std::move(managedBy);
804         asyncResp->res.jsonValue["EthernetInterfaces"]["@odata.id"] =
805             "/redfish/v1/Systems/hypervisor/EthernetInterfaces";
806         getHypervisorState(asyncResp);
807         getHypervisorActions(asyncResp);
808         // TODO: Add "SystemType" : "hypervisor"
809         });
810 }
811 
812 inline void handleHypervisorEthernetInterfacePatch(
813     App& app, const crow::Request& req,
814     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
815     const std::string& ifaceId)
816 {
817     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
818     {
819         return;
820     }
821     std::optional<std::string> hostName;
822     std::optional<std::vector<nlohmann::json>> ipv4StaticAddresses;
823     std::optional<nlohmann::json> ipv4Addresses;
824     std::optional<nlohmann::json> dhcpv4;
825     std::optional<bool> ipv4DHCPEnabled;
826 
827     if (!json_util::readJsonPatch(req, asyncResp->res, "HostName", hostName,
828                                   "IPv4StaticAddresses", ipv4StaticAddresses,
829                                   "IPv4Addresses", ipv4Addresses, "DHCPv4",
830                                   dhcpv4))
831     {
832         return;
833     }
834 
835     if (ipv4Addresses)
836     {
837         messages::propertyNotWritable(asyncResp->res, "IPv4Addresses");
838         return;
839     }
840 
841     if (dhcpv4)
842     {
843         if (!json_util::readJson(*dhcpv4, asyncResp->res, "DHCPEnabled",
844                                  ipv4DHCPEnabled))
845         {
846             return;
847         }
848     }
849 
850     getHypervisorIfaceData(
851         ifaceId,
852         [asyncResp, ifaceId, hostName = std::move(hostName),
853          ipv4StaticAddresses = std::move(ipv4StaticAddresses), ipv4DHCPEnabled,
854          dhcpv4 = std::move(dhcpv4)](bool success,
855                                      const EthernetInterfaceData& ethData,
856                                      const std::vector<IPv4AddressData>&) {
857         if (!success)
858         {
859             messages::resourceNotFound(asyncResp->res, "EthernetInterface",
860                                        ifaceId);
861             return;
862         }
863 
864         if (ipv4StaticAddresses)
865         {
866             const nlohmann::json& ipv4Static = *ipv4StaticAddresses;
867             if (ipv4Static.begin() == ipv4Static.end())
868             {
869                 messages::propertyValueTypeError(asyncResp->res, ipv4Static,
870                                                  "IPv4StaticAddresses");
871                 return;
872             }
873 
874             // One and only one hypervisor instance supported
875             if (ipv4Static.size() != 1)
876             {
877                 messages::propertyValueFormatError(asyncResp->res, ipv4Static,
878                                                    "IPv4StaticAddresses");
879                 return;
880             }
881 
882             const nlohmann::json& ipv4Json = ipv4Static[0];
883             // Check if the param is 'null'. If its null, it means
884             // that user wants to delete the IP address. Deleting
885             // the IP address is allowed only if its statically
886             // configured. Deleting the address originated from DHCP
887             // is not allowed.
888             if ((ipv4Json.is_null()) &&
889                 (translateDhcpEnabledToBool(ethData.dhcpEnabled, true)))
890             {
891                 BMCWEB_LOG_INFO << "Ignoring the delete on ipv4StaticAddresses "
892                                    "as the interface is DHCP enabled";
893             }
894             else
895             {
896                 handleHypervisorIPv4StaticPatch(ifaceId, ipv4Static, asyncResp);
897             }
898         }
899 
900         if (hostName)
901         {
902             handleHypervisorHostnamePatch(*hostName, asyncResp);
903         }
904 
905         if (dhcpv4)
906         {
907             setDHCPEnabled(ifaceId, *ipv4DHCPEnabled, asyncResp);
908         }
909 
910         // Set this interface to disabled/inactive. This will be set
911         // to enabled/active by the pldm once the hypervisor
912         // consumes the updated settings from the user.
913         setIPv4InterfaceEnabled(ifaceId, false, asyncResp);
914         });
915     asyncResp->res.result(boost::beast::http::status::accepted);
916 }
917 
918 inline void handleHypervisorResetActionGet(
919     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
920 {
921     // Only return action info if hypervisor D-Bus object present
922     constexpr std::array<std::string_view, 1> interfaces = {
923         "xyz.openbmc_project.State.Host"};
924     dbus::utility::getDbusObject(
925         "/xyz/openbmc_project/state/hypervisor0", interfaces,
926         [asyncResp](
927             const boost::system::error_code& ec,
928             const std::vector<std::pair<std::string, std::vector<std::string>>>&
929                 objInfo) {
930         if (ec)
931         {
932             BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
933 
934             // No hypervisor objects found by mapper
935             if (ec.value() == boost::system::errc::io_error)
936             {
937                 messages::resourceNotFound(asyncResp->res, "hypervisor",
938                                            "ResetActionInfo");
939                 return;
940             }
941 
942             messages::internalError(asyncResp->res);
943             return;
944         }
945 
946         // One and only one hypervisor instance supported
947         if (objInfo.size() != 1)
948         {
949             messages::internalError(asyncResp->res);
950             return;
951         }
952 
953         // The hypervisor object only support the ability to
954         // turn On The system object Action should be utilized
955         // for other operations
956 
957         asyncResp->res.jsonValue["@odata.type"] =
958             "#ActionInfo.v1_1_2.ActionInfo";
959         asyncResp->res.jsonValue["@odata.id"] =
960             "/redfish/v1/Systems/hypervisor/ResetActionInfo";
961         asyncResp->res.jsonValue["Name"] = "Reset Action Info";
962         asyncResp->res.jsonValue["Id"] = "ResetActionInfo";
963         nlohmann::json::array_t parameters;
964         nlohmann::json::object_t parameter;
965         parameter["Name"] = "ResetType";
966         parameter["Required"] = true;
967         parameter["DataType"] = "String";
968         nlohmann::json::array_t allowed;
969         allowed.emplace_back("On");
970         parameter["AllowableValues"] = std::move(allowed);
971         parameters.emplace_back(std::move(parameter));
972         asyncResp->res.jsonValue["Parameters"] = std::move(parameters);
973         });
974 }
975 
976 inline void handleHypervisorSystemResetPost(
977     App& app, const crow::Request& req,
978     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
979 {
980     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
981     {
982         return;
983     }
984     std::optional<std::string> resetType;
985     if (!json_util::readJsonAction(req, asyncResp->res, "ResetType", resetType))
986     {
987         // readJson adds appropriate error to response
988         return;
989     }
990 
991     if (!resetType)
992     {
993         messages::actionParameterMissing(asyncResp->res, "ComputerSystem.Reset",
994                                          "ResetType");
995         return;
996     }
997 
998     // Hypervisor object only support On operation
999     if (resetType != "On")
1000     {
1001         messages::propertyValueNotInList(asyncResp->res, *resetType,
1002                                          "ResetType");
1003         return;
1004     }
1005 
1006     std::string command = "xyz.openbmc_project.State.Host.Transition.On";
1007 
1008     sdbusplus::asio::setProperty(
1009         *crow::connections::systemBus, "xyz.openbmc_project.State.Hypervisor",
1010         "/xyz/openbmc_project/state/hypervisor0",
1011         "xyz.openbmc_project.State.Host", "RequestedHostTransition", command,
1012         [asyncResp, resetType](const boost::system::error_code& ec) {
1013         if (ec)
1014         {
1015             BMCWEB_LOG_ERROR << "D-Bus responses error: " << ec;
1016             if (ec.value() == boost::asio::error::invalid_argument)
1017             {
1018                 messages::actionParameterNotSupported(asyncResp->res,
1019                                                       *resetType, "Reset");
1020                 return;
1021             }
1022 
1023             if (ec.value() == boost::asio::error::host_unreachable)
1024             {
1025                 messages::resourceNotFound(asyncResp->res, "Actions", "Reset");
1026                 return;
1027             }
1028 
1029             messages::internalError(asyncResp->res);
1030             return;
1031         }
1032         messages::success(asyncResp->res);
1033         });
1034 }
1035 
1036 inline void requestRoutesHypervisorSystems(App& app)
1037 {
1038     /**
1039      * HypervisorInterfaceCollection class to handle the GET and PATCH on
1040      * Hypervisor Interface
1041      */
1042 
1043     BMCWEB_ROUTE(app, "/redfish/v1/Systems/hypervisor/EthernetInterfaces/")
1044         .privileges(redfish::privileges::getEthernetInterfaceCollection)
1045         .methods(boost::beast::http::verb::get)(std::bind_front(
1046             handleHypervisorEthernetInterfaceCollectionGet, std::ref(app)));
1047 
1048     BMCWEB_ROUTE(app,
1049                  "/redfish/v1/Systems/hypervisor/EthernetInterfaces/<str>/")
1050         .privileges(redfish::privileges::getEthernetInterface)
1051         .methods(boost::beast::http::verb::get)(std::bind_front(
1052             handleHypervisorEthernetInterfaceGet, std::ref(app)));
1053 
1054     BMCWEB_ROUTE(app,
1055                  "/redfish/v1/Systems/hypervisor/EthernetInterfaces/<str>/")
1056         .privileges(redfish::privileges::patchEthernetInterface)
1057         .methods(boost::beast::http::verb::patch)(std::bind_front(
1058             handleHypervisorEthernetInterfacePatch, std::ref(app)));
1059 
1060     BMCWEB_ROUTE(app,
1061                  "/redfish/v1/Systems/hypervisor/Actions/ComputerSystem.Reset/")
1062         .privileges(redfish::privileges::postComputerSystem)
1063         .methods(boost::beast::http::verb::post)(
1064             std::bind_front(handleHypervisorSystemResetPost, std::ref(app)));
1065 }
1066 } // namespace redfish
1067