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_singleton.hpp" 22 #include "dbus_utility.hpp" 23 #include "error_messages.hpp" 24 #include "health.hpp" 25 #include "human_sort.hpp" 26 #include "query.hpp" 27 #include "registries/privilege_registry.hpp" 28 #include "utils/ip_utils.hpp" 29 #include "utils/json_utils.hpp" 30 31 #include <boost/algorithm/string/classification.hpp> 32 #include <boost/algorithm/string/split.hpp> 33 #include <boost/url/format.hpp> 34 35 #include <array> 36 #include <optional> 37 #include <regex> 38 #include <string_view> 39 #include <vector> 40 41 namespace redfish 42 { 43 44 enum class LinkType 45 { 46 Local, 47 Global 48 }; 49 50 /** 51 * Structure for keeping IPv4 data required by Redfish 52 */ 53 struct IPv4AddressData 54 { 55 std::string id; 56 std::string address; 57 std::string domain; 58 std::string gateway; 59 std::string netmask; 60 std::string origin; 61 LinkType linktype{}; 62 bool isActive{}; 63 }; 64 65 /** 66 * Structure for keeping IPv6 data required by Redfish 67 */ 68 struct IPv6AddressData 69 { 70 std::string id; 71 std::string address; 72 std::string origin; 73 uint8_t prefixLength = 0; 74 }; 75 /** 76 * Structure for keeping basic single Ethernet Interface information 77 * available from DBus 78 */ 79 struct EthernetInterfaceData 80 { 81 uint32_t speed; 82 size_t mtuSize; 83 bool autoNeg; 84 bool dnsEnabled; 85 bool ntpEnabled; 86 bool hostNameEnabled; 87 bool linkUp; 88 bool nicEnabled; 89 std::string dhcpEnabled; 90 std::string operatingMode; 91 std::string hostName; 92 std::string defaultGateway; 93 std::string ipv6DefaultGateway; 94 std::string macAddress; 95 std::optional<uint32_t> vlanId; 96 std::vector<std::string> nameServers; 97 std::vector<std::string> staticNameServers; 98 std::vector<std::string> domainnames; 99 }; 100 101 struct DHCPParameters 102 { 103 std::optional<bool> dhcpv4Enabled; 104 std::optional<bool> useDnsServers; 105 std::optional<bool> useNtpServers; 106 std::optional<bool> useDomainName; 107 std::optional<std::string> dhcpv6OperatingMode; 108 }; 109 110 // Helper function that changes bits netmask notation (i.e. /24) 111 // into full dot notation 112 inline std::string getNetmask(unsigned int bits) 113 { 114 uint32_t value = 0xffffffff << (32 - bits); 115 std::string netmask = std::to_string((value >> 24) & 0xff) + "." + 116 std::to_string((value >> 16) & 0xff) + "." + 117 std::to_string((value >> 8) & 0xff) + "." + 118 std::to_string(value & 0xff); 119 return netmask; 120 } 121 122 inline bool translateDhcpEnabledToBool(const std::string& inputDHCP, 123 bool isIPv4) 124 { 125 if (isIPv4) 126 { 127 return ( 128 (inputDHCP == 129 "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.v4") || 130 (inputDHCP == 131 "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.both")); 132 } 133 return ((inputDHCP == 134 "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.v6") || 135 (inputDHCP == 136 "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.both")); 137 } 138 139 inline std::string getDhcpEnabledEnumeration(bool isIPv4, bool isIPv6) 140 { 141 if (isIPv4 && isIPv6) 142 { 143 return "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.both"; 144 } 145 if (isIPv4) 146 { 147 return "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.v4"; 148 } 149 if (isIPv6) 150 { 151 return "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.v6"; 152 } 153 return "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.none"; 154 } 155 156 inline std::string 157 translateAddressOriginDbusToRedfish(const std::string& inputOrigin, 158 bool isIPv4) 159 { 160 if (inputOrigin == "xyz.openbmc_project.Network.IP.AddressOrigin.Static") 161 { 162 return "Static"; 163 } 164 if (inputOrigin == "xyz.openbmc_project.Network.IP.AddressOrigin.LinkLocal") 165 { 166 if (isIPv4) 167 { 168 return "IPv4LinkLocal"; 169 } 170 return "LinkLocal"; 171 } 172 if (inputOrigin == "xyz.openbmc_project.Network.IP.AddressOrigin.DHCP") 173 { 174 if (isIPv4) 175 { 176 return "DHCP"; 177 } 178 return "DHCPv6"; 179 } 180 if (inputOrigin == "xyz.openbmc_project.Network.IP.AddressOrigin.SLAAC") 181 { 182 return "SLAAC"; 183 } 184 return ""; 185 } 186 187 inline bool extractEthernetInterfaceData( 188 const std::string& ethifaceId, 189 const dbus::utility::ManagedObjectType& dbusData, 190 EthernetInterfaceData& ethData) 191 { 192 bool idFound = false; 193 for (const auto& objpath : dbusData) 194 { 195 for (const auto& ifacePair : objpath.second) 196 { 197 if (objpath.first == "/xyz/openbmc_project/network/" + ethifaceId) 198 { 199 idFound = true; 200 if (ifacePair.first == "xyz.openbmc_project.Network.MACAddress") 201 { 202 for (const auto& propertyPair : ifacePair.second) 203 { 204 if (propertyPair.first == "MACAddress") 205 { 206 const std::string* mac = 207 std::get_if<std::string>(&propertyPair.second); 208 if (mac != nullptr) 209 { 210 ethData.macAddress = *mac; 211 } 212 } 213 } 214 } 215 else if (ifacePair.first == "xyz.openbmc_project.Network.VLAN") 216 { 217 for (const auto& propertyPair : ifacePair.second) 218 { 219 if (propertyPair.first == "Id") 220 { 221 const uint32_t* id = 222 std::get_if<uint32_t>(&propertyPair.second); 223 if (id != nullptr) 224 { 225 ethData.vlanId = *id; 226 } 227 } 228 } 229 } 230 else if (ifacePair.first == 231 "xyz.openbmc_project.Network.EthernetInterface") 232 { 233 for (const auto& propertyPair : ifacePair.second) 234 { 235 if (propertyPair.first == "AutoNeg") 236 { 237 const bool* autoNeg = 238 std::get_if<bool>(&propertyPair.second); 239 if (autoNeg != nullptr) 240 { 241 ethData.autoNeg = *autoNeg; 242 } 243 } 244 else if (propertyPair.first == "Speed") 245 { 246 const uint32_t* speed = 247 std::get_if<uint32_t>(&propertyPair.second); 248 if (speed != nullptr) 249 { 250 ethData.speed = *speed; 251 } 252 } 253 else if (propertyPair.first == "MTU") 254 { 255 const uint32_t* mtuSize = 256 std::get_if<uint32_t>(&propertyPair.second); 257 if (mtuSize != nullptr) 258 { 259 ethData.mtuSize = *mtuSize; 260 } 261 } 262 else if (propertyPair.first == "LinkUp") 263 { 264 const bool* linkUp = 265 std::get_if<bool>(&propertyPair.second); 266 if (linkUp != nullptr) 267 { 268 ethData.linkUp = *linkUp; 269 } 270 } 271 else if (propertyPair.first == "NICEnabled") 272 { 273 const bool* nicEnabled = 274 std::get_if<bool>(&propertyPair.second); 275 if (nicEnabled != nullptr) 276 { 277 ethData.nicEnabled = *nicEnabled; 278 } 279 } 280 else if (propertyPair.first == "Nameservers") 281 { 282 const std::vector<std::string>* nameservers = 283 std::get_if<std::vector<std::string>>( 284 &propertyPair.second); 285 if (nameservers != nullptr) 286 { 287 ethData.nameServers = *nameservers; 288 } 289 } 290 else if (propertyPair.first == "StaticNameServers") 291 { 292 const std::vector<std::string>* staticNameServers = 293 std::get_if<std::vector<std::string>>( 294 &propertyPair.second); 295 if (staticNameServers != nullptr) 296 { 297 ethData.staticNameServers = *staticNameServers; 298 } 299 } 300 else if (propertyPair.first == "DHCPEnabled") 301 { 302 const std::string* dhcpEnabled = 303 std::get_if<std::string>(&propertyPair.second); 304 if (dhcpEnabled != nullptr) 305 { 306 ethData.dhcpEnabled = *dhcpEnabled; 307 } 308 } 309 else if (propertyPair.first == "DomainName") 310 { 311 const std::vector<std::string>* domainNames = 312 std::get_if<std::vector<std::string>>( 313 &propertyPair.second); 314 if (domainNames != nullptr) 315 { 316 ethData.domainnames = *domainNames; 317 } 318 } 319 else if (propertyPair.first == "DefaultGateway") 320 { 321 const std::string* defaultGateway = 322 std::get_if<std::string>(&propertyPair.second); 323 if (defaultGateway != nullptr) 324 { 325 std::string defaultGatewayStr = *defaultGateway; 326 if (defaultGatewayStr.empty()) 327 { 328 ethData.defaultGateway = "0.0.0.0"; 329 } 330 else 331 { 332 ethData.defaultGateway = defaultGatewayStr; 333 } 334 } 335 } 336 else if (propertyPair.first == "DefaultGateway6") 337 { 338 const std::string* defaultGateway6 = 339 std::get_if<std::string>(&propertyPair.second); 340 if (defaultGateway6 != nullptr) 341 { 342 std::string defaultGateway6Str = 343 *defaultGateway6; 344 if (defaultGateway6Str.empty()) 345 { 346 ethData.ipv6DefaultGateway = 347 "0:0:0:0:0:0:0:0"; 348 } 349 else 350 { 351 ethData.ipv6DefaultGateway = 352 defaultGateway6Str; 353 } 354 } 355 } 356 } 357 } 358 } 359 360 if (objpath.first == "/xyz/openbmc_project/network/dhcp") 361 { 362 if (ifacePair.first == 363 "xyz.openbmc_project.Network.DHCPConfiguration") 364 { 365 for (const auto& propertyPair : ifacePair.second) 366 { 367 if (propertyPair.first == "DNSEnabled") 368 { 369 const bool* dnsEnabled = 370 std::get_if<bool>(&propertyPair.second); 371 if (dnsEnabled != nullptr) 372 { 373 ethData.dnsEnabled = *dnsEnabled; 374 } 375 } 376 else if (propertyPair.first == "NTPEnabled") 377 { 378 const bool* ntpEnabled = 379 std::get_if<bool>(&propertyPair.second); 380 if (ntpEnabled != nullptr) 381 { 382 ethData.ntpEnabled = *ntpEnabled; 383 } 384 } 385 else if (propertyPair.first == "HostNameEnabled") 386 { 387 const bool* hostNameEnabled = 388 std::get_if<bool>(&propertyPair.second); 389 if (hostNameEnabled != nullptr) 390 { 391 ethData.hostNameEnabled = *hostNameEnabled; 392 } 393 } 394 } 395 } 396 } 397 // System configuration shows up in the global namespace, so no need 398 // to check eth number 399 if (ifacePair.first == 400 "xyz.openbmc_project.Network.SystemConfiguration") 401 { 402 for (const auto& propertyPair : ifacePair.second) 403 { 404 if (propertyPair.first == "HostName") 405 { 406 const std::string* hostname = 407 std::get_if<std::string>(&propertyPair.second); 408 if (hostname != nullptr) 409 { 410 ethData.hostName = *hostname; 411 } 412 } 413 } 414 } 415 } 416 } 417 return idFound; 418 } 419 420 // Helper function that extracts data for single ethernet ipv6 address 421 inline void extractIPV6Data(const std::string& ethifaceId, 422 const dbus::utility::ManagedObjectType& dbusData, 423 std::vector<IPv6AddressData>& ipv6Config) 424 { 425 const std::string ipPathStart = "/xyz/openbmc_project/network/" + 426 ethifaceId; 427 428 // Since there might be several IPv6 configurations aligned with 429 // single ethernet interface, loop over all of them 430 for (const auto& objpath : dbusData) 431 { 432 // Check if proper pattern for object path appears 433 if (objpath.first.str.starts_with(ipPathStart + "/")) 434 { 435 for (const auto& interface : objpath.second) 436 { 437 if (interface.first == "xyz.openbmc_project.Network.IP") 438 { 439 auto type = std::find_if(interface.second.begin(), 440 interface.second.end(), 441 [](const auto& property) { 442 return property.first == "Type"; 443 }); 444 if (type == interface.second.end()) 445 { 446 continue; 447 } 448 449 const std::string* typeStr = 450 std::get_if<std::string>(&type->second); 451 452 if (typeStr == nullptr || 453 (*typeStr != 454 "xyz.openbmc_project.Network.IP.Protocol.IPv6")) 455 { 456 continue; 457 } 458 459 // Instance IPv6AddressData structure, and set as 460 // appropriate 461 IPv6AddressData& ipv6Address = ipv6Config.emplace_back(); 462 ipv6Address.id = 463 objpath.first.str.substr(ipPathStart.size()); 464 for (const auto& property : interface.second) 465 { 466 if (property.first == "Address") 467 { 468 const std::string* address = 469 std::get_if<std::string>(&property.second); 470 if (address != nullptr) 471 { 472 ipv6Address.address = *address; 473 } 474 } 475 else if (property.first == "Origin") 476 { 477 const std::string* origin = 478 std::get_if<std::string>(&property.second); 479 if (origin != nullptr) 480 { 481 ipv6Address.origin = 482 translateAddressOriginDbusToRedfish(*origin, 483 false); 484 } 485 } 486 else if (property.first == "PrefixLength") 487 { 488 const uint8_t* prefix = 489 std::get_if<uint8_t>(&property.second); 490 if (prefix != nullptr) 491 { 492 ipv6Address.prefixLength = *prefix; 493 } 494 } 495 else if (property.first == "Type" || 496 property.first == "Gateway") 497 { 498 // Type & Gateway is not used 499 } 500 else 501 { 502 BMCWEB_LOG_ERROR 503 << "Got extra property: " << property.first 504 << " on the " << objpath.first.str << " object"; 505 } 506 } 507 } 508 } 509 } 510 } 511 } 512 513 // Helper function that extracts data for single ethernet ipv4 address 514 inline void extractIPData(const std::string& ethifaceId, 515 const dbus::utility::ManagedObjectType& dbusData, 516 std::vector<IPv4AddressData>& ipv4Config) 517 { 518 const std::string ipPathStart = "/xyz/openbmc_project/network/" + 519 ethifaceId; 520 521 // Since there might be several IPv4 configurations aligned with 522 // single ethernet interface, loop over all of them 523 for (const auto& objpath : dbusData) 524 { 525 // Check if proper pattern for object path appears 526 if (objpath.first.str.starts_with(ipPathStart + "/")) 527 { 528 for (const auto& interface : objpath.second) 529 { 530 if (interface.first == "xyz.openbmc_project.Network.IP") 531 { 532 auto type = std::find_if(interface.second.begin(), 533 interface.second.end(), 534 [](const auto& property) { 535 return property.first == "Type"; 536 }); 537 if (type == interface.second.end()) 538 { 539 continue; 540 } 541 542 const std::string* typeStr = 543 std::get_if<std::string>(&type->second); 544 545 if (typeStr == nullptr || 546 (*typeStr != 547 "xyz.openbmc_project.Network.IP.Protocol.IPv4")) 548 { 549 continue; 550 } 551 552 // Instance IPv4AddressData structure, and set as 553 // appropriate 554 IPv4AddressData& ipv4Address = ipv4Config.emplace_back(); 555 ipv4Address.id = 556 objpath.first.str.substr(ipPathStart.size()); 557 for (const auto& property : interface.second) 558 { 559 if (property.first == "Address") 560 { 561 const std::string* address = 562 std::get_if<std::string>(&property.second); 563 if (address != nullptr) 564 { 565 ipv4Address.address = *address; 566 } 567 } 568 else if (property.first == "Origin") 569 { 570 const std::string* origin = 571 std::get_if<std::string>(&property.second); 572 if (origin != nullptr) 573 { 574 ipv4Address.origin = 575 translateAddressOriginDbusToRedfish(*origin, 576 true); 577 } 578 } 579 else if (property.first == "PrefixLength") 580 { 581 const uint8_t* mask = 582 std::get_if<uint8_t>(&property.second); 583 if (mask != nullptr) 584 { 585 // convert it to the string 586 ipv4Address.netmask = getNetmask(*mask); 587 } 588 } 589 else if (property.first == "Type" || 590 property.first == "Gateway") 591 { 592 // Type & Gateway is not used 593 } 594 else 595 { 596 BMCWEB_LOG_ERROR 597 << "Got extra property: " << property.first 598 << " on the " << objpath.first.str << " object"; 599 } 600 } 601 // Check if given address is local, or global 602 ipv4Address.linktype = 603 ipv4Address.address.starts_with("169.254.") 604 ? LinkType::Local 605 : LinkType::Global; 606 } 607 } 608 } 609 } 610 } 611 612 /** 613 * @brief Deletes given IPv4 interface 614 * 615 * @param[in] ifaceId Id of interface whose IP should be deleted 616 * @param[in] ipHash DBus Hash id of IP that should be deleted 617 * @param[io] asyncResp Response object that will be returned to client 618 * 619 * @return None 620 */ 621 inline void deleteIPAddress(const std::string& ifaceId, 622 const std::string& ipHash, 623 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 624 { 625 crow::connections::systemBus->async_method_call( 626 [asyncResp](const boost::system::error_code& ec) { 627 if (ec) 628 { 629 messages::internalError(asyncResp->res); 630 } 631 }, 632 "xyz.openbmc_project.Network", 633 "/xyz/openbmc_project/network/" + ifaceId + ipHash, 634 "xyz.openbmc_project.Object.Delete", "Delete"); 635 } 636 637 inline void updateIPv4DefaultGateway( 638 const std::string& ifaceId, const std::string& gateway, 639 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 640 { 641 sdbusplus::asio::setProperty( 642 *crow::connections::systemBus, "xyz.openbmc_project.Network", 643 "/xyz/openbmc_project/network/" + ifaceId, 644 "xyz.openbmc_project.Network.EthernetInterface", "DefaultGateway", 645 gateway, [asyncResp](const boost::system::error_code& ec) { 646 if (ec) 647 { 648 messages::internalError(asyncResp->res); 649 return; 650 } 651 asyncResp->res.result(boost::beast::http::status::no_content); 652 }); 653 } 654 /** 655 * @brief Creates a static IPv4 entry 656 * 657 * @param[in] ifaceId Id of interface upon which to create the IPv4 entry 658 * @param[in] prefixLength IPv4 prefix syntax for the subnet mask 659 * @param[in] gateway IPv4 address of this interfaces gateway 660 * @param[in] address IPv4 address to assign to this interface 661 * @param[io] asyncResp Response object that will be returned to client 662 * 663 * @return None 664 */ 665 inline void createIPv4(const std::string& ifaceId, uint8_t prefixLength, 666 const std::string& gateway, const std::string& address, 667 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 668 { 669 auto createIpHandler = 670 [asyncResp, ifaceId, gateway](const boost::system::error_code& ec) { 671 if (ec) 672 { 673 messages::internalError(asyncResp->res); 674 return; 675 } 676 updateIPv4DefaultGateway(ifaceId, gateway, asyncResp); 677 }; 678 679 crow::connections::systemBus->async_method_call( 680 std::move(createIpHandler), "xyz.openbmc_project.Network", 681 "/xyz/openbmc_project/network/" + ifaceId, 682 "xyz.openbmc_project.Network.IP.Create", "IP", 683 "xyz.openbmc_project.Network.IP.Protocol.IPv4", address, prefixLength, 684 gateway); 685 } 686 687 /** 688 * @brief Deletes the IPv6 entry for this interface and creates a replacement 689 * static IPv6 entry 690 * 691 * @param[in] ifaceId Id of interface upon which to create the IPv6 entry 692 * @param[in] id The unique hash entry identifying the DBus entry 693 * @param[in] prefixLength IPv6 prefix syntax for the subnet mask 694 * @param[in] address IPv6 address to assign to this interface 695 * @param[io] asyncResp Response object that will be returned to client 696 * 697 * @return None 698 */ 699 700 enum class IpVersion 701 { 702 IpV4, 703 IpV6 704 }; 705 706 inline void deleteAndCreateIPAddress( 707 IpVersion version, const std::string& ifaceId, const std::string& id, 708 uint8_t prefixLength, const std::string& address, 709 const std::string& gateway, 710 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 711 { 712 crow::connections::systemBus->async_method_call( 713 [asyncResp, version, ifaceId, address, prefixLength, 714 gateway](const boost::system::error_code& ec) { 715 if (ec) 716 { 717 messages::internalError(asyncResp->res); 718 } 719 std::string protocol = "xyz.openbmc_project.Network.IP.Protocol."; 720 protocol += version == IpVersion::IpV4 ? "IPv4" : "IPv6"; 721 crow::connections::systemBus->async_method_call( 722 [asyncResp](const boost::system::error_code& ec2) { 723 if (ec2) 724 { 725 messages::internalError(asyncResp->res); 726 } 727 }, 728 "xyz.openbmc_project.Network", 729 "/xyz/openbmc_project/network/" + ifaceId, 730 "xyz.openbmc_project.Network.IP.Create", "IP", protocol, address, 731 prefixLength, gateway); 732 }, 733 "xyz.openbmc_project.Network", 734 "/xyz/openbmc_project/network/" + ifaceId + id, 735 "xyz.openbmc_project.Object.Delete", "Delete"); 736 } 737 738 /** 739 * @brief Creates IPv6 with given data 740 * 741 * @param[in] ifaceId Id of interface whose IP should be added 742 * @param[in] prefixLength Prefix length that needs to be added 743 * @param[in] address IP address that needs to be added 744 * @param[io] asyncResp Response object that will be returned to client 745 * 746 * @return None 747 */ 748 inline void createIPv6(const std::string& ifaceId, uint8_t prefixLength, 749 const std::string& address, 750 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 751 { 752 auto createIpHandler = [asyncResp](const boost::system::error_code& ec) { 753 if (ec) 754 { 755 messages::internalError(asyncResp->res); 756 } 757 }; 758 // Passing null for gateway, as per redfish spec IPv6StaticAddresses object 759 // does not have associated gateway property 760 crow::connections::systemBus->async_method_call( 761 std::move(createIpHandler), "xyz.openbmc_project.Network", 762 "/xyz/openbmc_project/network/" + ifaceId, 763 "xyz.openbmc_project.Network.IP.Create", "IP", 764 "xyz.openbmc_project.Network.IP.Protocol.IPv6", address, prefixLength, 765 ""); 766 } 767 768 /** 769 * Function that retrieves all properties for given Ethernet Interface 770 * Object 771 * from EntityManager Network Manager 772 * @param ethiface_id a eth interface id to query on DBus 773 * @param callback a function that shall be called to convert Dbus output 774 * into JSON 775 */ 776 template <typename CallbackFunc> 777 void getEthernetIfaceData(const std::string& ethifaceId, 778 CallbackFunc&& callback) 779 { 780 sdbusplus::message::object_path path("/xyz/openbmc_project/network"); 781 dbus::utility::getManagedObjects( 782 "xyz.openbmc_project.Network", path, 783 [ethifaceId{std::string{ethifaceId}}, 784 callback{std::forward<CallbackFunc>(callback)}]( 785 const boost::system::error_code& ec, 786 const dbus::utility::ManagedObjectType& resp) { 787 EthernetInterfaceData ethData{}; 788 std::vector<IPv4AddressData> ipv4Data; 789 std::vector<IPv6AddressData> ipv6Data; 790 791 if (ec) 792 { 793 callback(false, ethData, ipv4Data, ipv6Data); 794 return; 795 } 796 797 bool found = extractEthernetInterfaceData(ethifaceId, resp, ethData); 798 if (!found) 799 { 800 callback(false, ethData, ipv4Data, ipv6Data); 801 return; 802 } 803 804 extractIPData(ethifaceId, resp, ipv4Data); 805 // Fix global GW 806 for (IPv4AddressData& ipv4 : ipv4Data) 807 { 808 if (((ipv4.linktype == LinkType::Global) && 809 (ipv4.gateway == "0.0.0.0")) || 810 (ipv4.origin == "DHCP") || (ipv4.origin == "Static")) 811 { 812 ipv4.gateway = ethData.defaultGateway; 813 } 814 } 815 816 extractIPV6Data(ethifaceId, resp, ipv6Data); 817 // Finally make a callback with useful data 818 callback(true, ethData, ipv4Data, ipv6Data); 819 }); 820 } 821 822 /** 823 * Function that retrieves all Ethernet Interfaces available through Network 824 * Manager 825 * @param callback a function that shall be called to convert Dbus output 826 * into JSON. 827 */ 828 template <typename CallbackFunc> 829 void getEthernetIfaceList(CallbackFunc&& callback) 830 { 831 sdbusplus::message::object_path path("/xyz/openbmc_project/network"); 832 dbus::utility::getManagedObjects( 833 "xyz.openbmc_project.Network", path, 834 [callback{std::forward<CallbackFunc>(callback)}]( 835 const boost::system::error_code& ec, 836 const dbus::utility::ManagedObjectType& resp) { 837 // Callback requires vector<string> to retrieve all available 838 // ethernet interfaces 839 std::vector<std::string> ifaceList; 840 ifaceList.reserve(resp.size()); 841 if (ec) 842 { 843 callback(false, ifaceList); 844 return; 845 } 846 847 // Iterate over all retrieved ObjectPaths. 848 for (const auto& objpath : resp) 849 { 850 // And all interfaces available for certain ObjectPath. 851 for (const auto& interface : objpath.second) 852 { 853 // If interface is 854 // xyz.openbmc_project.Network.EthernetInterface, this is 855 // what we're looking for. 856 if (interface.first == 857 "xyz.openbmc_project.Network.EthernetInterface") 858 { 859 std::string ifaceId = objpath.first.filename(); 860 if (ifaceId.empty()) 861 { 862 continue; 863 } 864 // and put it into output vector. 865 ifaceList.emplace_back(ifaceId); 866 } 867 } 868 } 869 870 std::sort(ifaceList.begin(), ifaceList.end(), 871 AlphanumLess<std::string>()); 872 873 // Finally make a callback with useful data 874 callback(true, ifaceList); 875 }); 876 } 877 878 inline void 879 handleHostnamePatch(const std::string& hostname, 880 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 881 { 882 // SHOULD handle host names of up to 255 characters(RFC 1123) 883 if (hostname.length() > 255) 884 { 885 messages::propertyValueFormatError(asyncResp->res, hostname, 886 "HostName"); 887 return; 888 } 889 sdbusplus::asio::setProperty( 890 *crow::connections::systemBus, "xyz.openbmc_project.Network", 891 "/xyz/openbmc_project/network/config", 892 "xyz.openbmc_project.Network.SystemConfiguration", "HostName", hostname, 893 [asyncResp](const boost::system::error_code& ec) { 894 if (ec) 895 { 896 messages::internalError(asyncResp->res); 897 } 898 }); 899 } 900 901 inline void 902 handleMTUSizePatch(const std::string& ifaceId, const size_t mtuSize, 903 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 904 { 905 sdbusplus::message::object_path objPath = "/xyz/openbmc_project/network/" + 906 ifaceId; 907 sdbusplus::asio::setProperty( 908 *crow::connections::systemBus, "xyz.openbmc_project.Network", objPath, 909 "xyz.openbmc_project.Network.EthernetInterface", "MTU", mtuSize, 910 [asyncResp](const boost::system::error_code& ec) { 911 if (ec) 912 { 913 messages::internalError(asyncResp->res); 914 } 915 }); 916 } 917 918 inline void 919 handleDomainnamePatch(const std::string& ifaceId, 920 const std::string& domainname, 921 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 922 { 923 std::vector<std::string> vectorDomainname = {domainname}; 924 sdbusplus::asio::setProperty( 925 *crow::connections::systemBus, "xyz.openbmc_project.Network", 926 "/xyz/openbmc_project/network/" + ifaceId, 927 "xyz.openbmc_project.Network.EthernetInterface", "DomainName", 928 vectorDomainname, [asyncResp](const boost::system::error_code& ec) { 929 if (ec) 930 { 931 messages::internalError(asyncResp->res); 932 } 933 }); 934 } 935 936 inline bool isHostnameValid(const std::string& hostname) 937 { 938 // A valid host name can never have the dotted-decimal form (RFC 1123) 939 if (std::all_of(hostname.begin(), hostname.end(), ::isdigit)) 940 { 941 return false; 942 } 943 // Each label(hostname/subdomains) within a valid FQDN 944 // MUST handle host names of up to 63 characters (RFC 1123) 945 // labels cannot start or end with hyphens (RFC 952) 946 // labels can start with numbers (RFC 1123) 947 const static std::regex pattern( 948 "^[a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\\-]{0,61}[a-zA-Z0-9]$"); 949 950 return std::regex_match(hostname, pattern); 951 } 952 953 inline bool isDomainnameValid(const std::string& domainname) 954 { 955 // Can have multiple subdomains 956 // Top Level Domain's min length is 2 character 957 const static std::regex pattern( 958 "^([A-Za-z0-9][a-zA-Z0-9\\-]{1,61}|[a-zA-Z0-9]{1,30}\\.)*[a-zA-Z]{2,}$"); 959 960 return std::regex_match(domainname, pattern); 961 } 962 963 inline void handleFqdnPatch(const std::string& ifaceId, const std::string& fqdn, 964 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 965 { 966 // Total length of FQDN must not exceed 255 characters(RFC 1035) 967 if (fqdn.length() > 255) 968 { 969 messages::propertyValueFormatError(asyncResp->res, fqdn, "FQDN"); 970 return; 971 } 972 973 size_t pos = fqdn.find('.'); 974 if (pos == std::string::npos) 975 { 976 messages::propertyValueFormatError(asyncResp->res, fqdn, "FQDN"); 977 return; 978 } 979 980 std::string hostname; 981 std::string domainname; 982 domainname = (fqdn).substr(pos + 1); 983 hostname = (fqdn).substr(0, pos); 984 985 if (!isHostnameValid(hostname) || !isDomainnameValid(domainname)) 986 { 987 messages::propertyValueFormatError(asyncResp->res, fqdn, "FQDN"); 988 return; 989 } 990 991 handleHostnamePatch(hostname, asyncResp); 992 handleDomainnamePatch(ifaceId, domainname, asyncResp); 993 } 994 995 inline void 996 handleMACAddressPatch(const std::string& ifaceId, 997 const std::string& macAddress, 998 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 999 { 1000 static constexpr std::string_view dbusNotAllowedError = 1001 "xyz.openbmc_project.Common.Error.NotAllowed"; 1002 1003 sdbusplus::asio::setProperty( 1004 *crow::connections::systemBus, "xyz.openbmc_project.Network", 1005 "/xyz/openbmc_project/network/" + ifaceId, 1006 "xyz.openbmc_project.Network.MACAddress", "MACAddress", macAddress, 1007 [asyncResp](const boost::system::error_code& ec, 1008 const sdbusplus::message_t& msg) { 1009 if (ec) 1010 { 1011 const sd_bus_error* err = msg.get_error(); 1012 if (err == nullptr) 1013 { 1014 messages::internalError(asyncResp->res); 1015 return; 1016 } 1017 if (err->name == dbusNotAllowedError) 1018 { 1019 messages::propertyNotWritable(asyncResp->res, "MACAddress"); 1020 return; 1021 } 1022 messages::internalError(asyncResp->res); 1023 return; 1024 } 1025 }); 1026 } 1027 1028 inline void setDHCPEnabled(const std::string& ifaceId, 1029 const std::string& propertyName, const bool v4Value, 1030 const bool v6Value, 1031 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 1032 { 1033 const std::string dhcp = getDhcpEnabledEnumeration(v4Value, v6Value); 1034 sdbusplus::asio::setProperty( 1035 *crow::connections::systemBus, "xyz.openbmc_project.Network", 1036 "/xyz/openbmc_project/network/" + ifaceId, 1037 "xyz.openbmc_project.Network.EthernetInterface", propertyName, dhcp, 1038 [asyncResp](const boost::system::error_code& ec) { 1039 if (ec) 1040 { 1041 BMCWEB_LOG_ERROR << "D-Bus responses error: " << ec; 1042 messages::internalError(asyncResp->res); 1043 return; 1044 } 1045 messages::success(asyncResp->res); 1046 }); 1047 } 1048 1049 inline void setEthernetInterfaceBoolProperty( 1050 const std::string& ifaceId, const std::string& propertyName, 1051 const bool& value, const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 1052 { 1053 sdbusplus::asio::setProperty( 1054 *crow::connections::systemBus, "xyz.openbmc_project.Network", 1055 "/xyz/openbmc_project/network/" + ifaceId, 1056 "xyz.openbmc_project.Network.EthernetInterface", propertyName, value, 1057 [asyncResp](const boost::system::error_code& ec) { 1058 if (ec) 1059 { 1060 BMCWEB_LOG_ERROR << "D-Bus responses error: " << ec; 1061 messages::internalError(asyncResp->res); 1062 return; 1063 } 1064 }); 1065 } 1066 1067 inline void setDHCPv4Config(const std::string& propertyName, const bool& value, 1068 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 1069 { 1070 BMCWEB_LOG_DEBUG << propertyName << " = " << value; 1071 sdbusplus::asio::setProperty( 1072 *crow::connections::systemBus, "xyz.openbmc_project.Network", 1073 "/xyz/openbmc_project/network/dhcp", 1074 "xyz.openbmc_project.Network.DHCPConfiguration", propertyName, value, 1075 [asyncResp](const boost::system::error_code& ec) { 1076 if (ec) 1077 { 1078 BMCWEB_LOG_ERROR << "D-Bus responses error: " << ec; 1079 messages::internalError(asyncResp->res); 1080 return; 1081 } 1082 }); 1083 } 1084 1085 inline void handleDHCPPatch(const std::string& ifaceId, 1086 const EthernetInterfaceData& ethData, 1087 const DHCPParameters& v4dhcpParms, 1088 const DHCPParameters& v6dhcpParms, 1089 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 1090 { 1091 bool ipv4Active = translateDhcpEnabledToBool(ethData.dhcpEnabled, true); 1092 bool ipv6Active = translateDhcpEnabledToBool(ethData.dhcpEnabled, false); 1093 1094 bool nextv4DHCPState = 1095 v4dhcpParms.dhcpv4Enabled ? *v4dhcpParms.dhcpv4Enabled : ipv4Active; 1096 1097 bool nextv6DHCPState{}; 1098 if (v6dhcpParms.dhcpv6OperatingMode) 1099 { 1100 if ((*v6dhcpParms.dhcpv6OperatingMode != "Stateful") && 1101 (*v6dhcpParms.dhcpv6OperatingMode != "Stateless") && 1102 (*v6dhcpParms.dhcpv6OperatingMode != "Disabled")) 1103 { 1104 messages::propertyValueFormatError(asyncResp->res, 1105 *v6dhcpParms.dhcpv6OperatingMode, 1106 "OperatingMode"); 1107 return; 1108 } 1109 nextv6DHCPState = (*v6dhcpParms.dhcpv6OperatingMode == "Stateful"); 1110 } 1111 else 1112 { 1113 nextv6DHCPState = ipv6Active; 1114 } 1115 1116 bool nextDNS{}; 1117 if (v4dhcpParms.useDnsServers && v6dhcpParms.useDnsServers) 1118 { 1119 if (*v4dhcpParms.useDnsServers != *v6dhcpParms.useDnsServers) 1120 { 1121 messages::generalError(asyncResp->res); 1122 return; 1123 } 1124 nextDNS = *v4dhcpParms.useDnsServers; 1125 } 1126 else if (v4dhcpParms.useDnsServers) 1127 { 1128 nextDNS = *v4dhcpParms.useDnsServers; 1129 } 1130 else if (v6dhcpParms.useDnsServers) 1131 { 1132 nextDNS = *v6dhcpParms.useDnsServers; 1133 } 1134 else 1135 { 1136 nextDNS = ethData.dnsEnabled; 1137 } 1138 1139 bool nextNTP{}; 1140 if (v4dhcpParms.useNtpServers && v6dhcpParms.useNtpServers) 1141 { 1142 if (*v4dhcpParms.useNtpServers != *v6dhcpParms.useNtpServers) 1143 { 1144 messages::generalError(asyncResp->res); 1145 return; 1146 } 1147 nextNTP = *v4dhcpParms.useNtpServers; 1148 } 1149 else if (v4dhcpParms.useNtpServers) 1150 { 1151 nextNTP = *v4dhcpParms.useNtpServers; 1152 } 1153 else if (v6dhcpParms.useNtpServers) 1154 { 1155 nextNTP = *v6dhcpParms.useNtpServers; 1156 } 1157 else 1158 { 1159 nextNTP = ethData.ntpEnabled; 1160 } 1161 1162 bool nextUseDomain{}; 1163 if (v4dhcpParms.useDomainName && v6dhcpParms.useDomainName) 1164 { 1165 if (*v4dhcpParms.useDomainName != *v6dhcpParms.useDomainName) 1166 { 1167 messages::generalError(asyncResp->res); 1168 return; 1169 } 1170 nextUseDomain = *v4dhcpParms.useDomainName; 1171 } 1172 else if (v4dhcpParms.useDomainName) 1173 { 1174 nextUseDomain = *v4dhcpParms.useDomainName; 1175 } 1176 else if (v6dhcpParms.useDomainName) 1177 { 1178 nextUseDomain = *v6dhcpParms.useDomainName; 1179 } 1180 else 1181 { 1182 nextUseDomain = ethData.hostNameEnabled; 1183 } 1184 1185 BMCWEB_LOG_DEBUG << "set DHCPEnabled..."; 1186 setDHCPEnabled(ifaceId, "DHCPEnabled", nextv4DHCPState, nextv6DHCPState, 1187 asyncResp); 1188 BMCWEB_LOG_DEBUG << "set DNSEnabled..."; 1189 setDHCPv4Config("DNSEnabled", nextDNS, asyncResp); 1190 BMCWEB_LOG_DEBUG << "set NTPEnabled..."; 1191 setDHCPv4Config("NTPEnabled", nextNTP, asyncResp); 1192 BMCWEB_LOG_DEBUG << "set HostNameEnabled..."; 1193 setDHCPv4Config("HostNameEnabled", nextUseDomain, asyncResp); 1194 } 1195 1196 inline std::vector<IPv4AddressData>::const_iterator getNextStaticIpEntry( 1197 const std::vector<IPv4AddressData>::const_iterator& head, 1198 const std::vector<IPv4AddressData>::const_iterator& end) 1199 { 1200 return std::find_if(head, end, [](const IPv4AddressData& value) { 1201 return value.origin == "Static"; 1202 }); 1203 } 1204 1205 inline std::vector<IPv6AddressData>::const_iterator getNextStaticIpEntry( 1206 const std::vector<IPv6AddressData>::const_iterator& head, 1207 const std::vector<IPv6AddressData>::const_iterator& end) 1208 { 1209 return std::find_if(head, end, [](const IPv6AddressData& value) { 1210 return value.origin == "Static"; 1211 }); 1212 } 1213 1214 inline void 1215 handleIPv4StaticPatch(const std::string& ifaceId, 1216 nlohmann::json::array_t& input, 1217 const std::vector<IPv4AddressData>& ipv4Data, 1218 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 1219 { 1220 if (input.empty()) 1221 { 1222 messages::propertyValueTypeError(asyncResp->res, input, 1223 "IPv4StaticAddresses"); 1224 return; 1225 } 1226 1227 unsigned entryIdx = 1; 1228 // Find the first static IP address currently active on the NIC and 1229 // match it to the first JSON element in the IPv4StaticAddresses array. 1230 // Match each subsequent JSON element to the next static IP programmed 1231 // into the NIC. 1232 std::vector<IPv4AddressData>::const_iterator nicIpEntry = 1233 getNextStaticIpEntry(ipv4Data.cbegin(), ipv4Data.cend()); 1234 1235 for (nlohmann::json& thisJson : input) 1236 { 1237 std::string pathString = "IPv4StaticAddresses/" + 1238 std::to_string(entryIdx); 1239 1240 if (!thisJson.is_null() && !thisJson.empty()) 1241 { 1242 std::optional<std::string> address; 1243 std::optional<std::string> subnetMask; 1244 std::optional<std::string> gateway; 1245 1246 if (!json_util::readJson(thisJson, asyncResp->res, "Address", 1247 address, "SubnetMask", subnetMask, 1248 "Gateway", gateway)) 1249 { 1250 messages::propertyValueFormatError(asyncResp->res, thisJson, 1251 pathString); 1252 return; 1253 } 1254 1255 // Find the address/subnet/gateway values. Any values that are 1256 // not explicitly provided are assumed to be unmodified from the 1257 // current state of the interface. Merge existing state into the 1258 // current request. 1259 const std::string* addr = nullptr; 1260 const std::string* gw = nullptr; 1261 uint8_t prefixLength = 0; 1262 bool errorInEntry = false; 1263 if (address) 1264 { 1265 if (ip_util::ipv4VerifyIpAndGetBitcount(*address)) 1266 { 1267 addr = &(*address); 1268 } 1269 else 1270 { 1271 messages::propertyValueFormatError(asyncResp->res, *address, 1272 pathString + "/Address"); 1273 errorInEntry = true; 1274 } 1275 } 1276 else if (nicIpEntry != ipv4Data.cend()) 1277 { 1278 addr = &(nicIpEntry->address); 1279 } 1280 else 1281 { 1282 messages::propertyMissing(asyncResp->res, 1283 pathString + "/Address"); 1284 errorInEntry = true; 1285 } 1286 1287 if (subnetMask) 1288 { 1289 if (!ip_util::ipv4VerifyIpAndGetBitcount(*subnetMask, 1290 &prefixLength)) 1291 { 1292 messages::propertyValueFormatError( 1293 asyncResp->res, *subnetMask, 1294 pathString + "/SubnetMask"); 1295 errorInEntry = true; 1296 } 1297 } 1298 else if (nicIpEntry != ipv4Data.cend()) 1299 { 1300 if (!ip_util::ipv4VerifyIpAndGetBitcount(nicIpEntry->netmask, 1301 &prefixLength)) 1302 { 1303 messages::propertyValueFormatError( 1304 asyncResp->res, nicIpEntry->netmask, 1305 pathString + "/SubnetMask"); 1306 errorInEntry = true; 1307 } 1308 } 1309 else 1310 { 1311 messages::propertyMissing(asyncResp->res, 1312 pathString + "/SubnetMask"); 1313 errorInEntry = true; 1314 } 1315 1316 if (gateway) 1317 { 1318 if (ip_util::ipv4VerifyIpAndGetBitcount(*gateway)) 1319 { 1320 gw = &(*gateway); 1321 } 1322 else 1323 { 1324 messages::propertyValueFormatError(asyncResp->res, *gateway, 1325 pathString + "/Gateway"); 1326 errorInEntry = true; 1327 } 1328 } 1329 else if (nicIpEntry != ipv4Data.cend()) 1330 { 1331 gw = &nicIpEntry->gateway; 1332 } 1333 else 1334 { 1335 messages::propertyMissing(asyncResp->res, 1336 pathString + "/Gateway"); 1337 errorInEntry = true; 1338 } 1339 1340 if (errorInEntry) 1341 { 1342 return; 1343 } 1344 1345 if (nicIpEntry != ipv4Data.cend()) 1346 { 1347 deleteAndCreateIPAddress(IpVersion::IpV4, ifaceId, 1348 nicIpEntry->id, prefixLength, *gw, 1349 *addr, asyncResp); 1350 nicIpEntry = getNextStaticIpEntry(++nicIpEntry, 1351 ipv4Data.cend()); 1352 } 1353 else 1354 { 1355 createIPv4(ifaceId, prefixLength, *gateway, *address, 1356 asyncResp); 1357 } 1358 entryIdx++; 1359 } 1360 else 1361 { 1362 if (nicIpEntry == ipv4Data.cend()) 1363 { 1364 // Requesting a DELETE/DO NOT MODIFY action for an item 1365 // that isn't present on the eth(n) interface. Input JSON is 1366 // in error, so bail out. 1367 if (thisJson.is_null()) 1368 { 1369 messages::resourceCannotBeDeleted(asyncResp->res); 1370 return; 1371 } 1372 messages::propertyValueFormatError(asyncResp->res, thisJson, 1373 pathString); 1374 return; 1375 } 1376 1377 if (thisJson.is_null()) 1378 { 1379 deleteIPAddress(ifaceId, nicIpEntry->id, asyncResp); 1380 } 1381 if (nicIpEntry != ipv4Data.cend()) 1382 { 1383 nicIpEntry = getNextStaticIpEntry(++nicIpEntry, 1384 ipv4Data.cend()); 1385 } 1386 entryIdx++; 1387 } 1388 } 1389 } 1390 1391 inline void handleStaticNameServersPatch( 1392 const std::string& ifaceId, 1393 const std::vector<std::string>& updatedStaticNameServers, 1394 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 1395 { 1396 sdbusplus::asio::setProperty( 1397 *crow::connections::systemBus, "xyz.openbmc_project.Network", 1398 "/xyz/openbmc_project/network/" + ifaceId, 1399 "xyz.openbmc_project.Network.EthernetInterface", "StaticNameServers", 1400 updatedStaticNameServers, 1401 [asyncResp](const boost::system::error_code& ec) { 1402 if (ec) 1403 { 1404 messages::internalError(asyncResp->res); 1405 return; 1406 } 1407 }); 1408 } 1409 1410 inline void handleIPv6StaticAddressesPatch( 1411 const std::string& ifaceId, const nlohmann::json::array_t& input, 1412 const std::vector<IPv6AddressData>& ipv6Data, 1413 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 1414 { 1415 if (input.empty()) 1416 { 1417 messages::propertyValueTypeError(asyncResp->res, input, 1418 "IPv6StaticAddresses"); 1419 return; 1420 } 1421 size_t entryIdx = 1; 1422 std::vector<IPv6AddressData>::const_iterator nicIpEntry = 1423 getNextStaticIpEntry(ipv6Data.cbegin(), ipv6Data.cend()); 1424 for (const nlohmann::json& thisJson : input) 1425 { 1426 std::string pathString = "IPv6StaticAddresses/" + 1427 std::to_string(entryIdx); 1428 1429 if (!thisJson.is_null() && !thisJson.empty()) 1430 { 1431 std::optional<std::string> address; 1432 std::optional<uint8_t> prefixLength; 1433 nlohmann::json thisJsonCopy = thisJson; 1434 if (!json_util::readJson(thisJsonCopy, asyncResp->res, "Address", 1435 address, "PrefixLength", prefixLength)) 1436 { 1437 messages::propertyValueFormatError(asyncResp->res, thisJson, 1438 pathString); 1439 return; 1440 } 1441 1442 const std::string* addr = nullptr; 1443 uint8_t prefix = 0; 1444 1445 // Find the address and prefixLength values. Any values that are 1446 // not explicitly provided are assumed to be unmodified from the 1447 // current state of the interface. Merge existing state into the 1448 // current request. 1449 if (address) 1450 { 1451 addr = &(*address); 1452 } 1453 else if (nicIpEntry != ipv6Data.end()) 1454 { 1455 addr = &(nicIpEntry->address); 1456 } 1457 else 1458 { 1459 messages::propertyMissing(asyncResp->res, 1460 pathString + "/Address"); 1461 return; 1462 } 1463 1464 if (prefixLength) 1465 { 1466 prefix = *prefixLength; 1467 } 1468 else if (nicIpEntry != ipv6Data.end()) 1469 { 1470 prefix = nicIpEntry->prefixLength; 1471 } 1472 else 1473 { 1474 messages::propertyMissing(asyncResp->res, 1475 pathString + "/PrefixLength"); 1476 return; 1477 } 1478 1479 if (nicIpEntry != ipv6Data.end()) 1480 { 1481 deleteAndCreateIPAddress(IpVersion::IpV6, ifaceId, 1482 nicIpEntry->id, prefix, "", *addr, 1483 asyncResp); 1484 nicIpEntry = getNextStaticIpEntry(++nicIpEntry, 1485 ipv6Data.cend()); 1486 } 1487 else 1488 { 1489 createIPv6(ifaceId, *prefixLength, *addr, asyncResp); 1490 } 1491 entryIdx++; 1492 } 1493 else 1494 { 1495 if (nicIpEntry == ipv6Data.end()) 1496 { 1497 // Requesting a DELETE/DO NOT MODIFY action for an item 1498 // that isn't present on the eth(n) interface. Input JSON is 1499 // in error, so bail out. 1500 if (thisJson.is_null()) 1501 { 1502 messages::resourceCannotBeDeleted(asyncResp->res); 1503 return; 1504 } 1505 messages::propertyValueFormatError(asyncResp->res, thisJson, 1506 pathString); 1507 return; 1508 } 1509 1510 if (thisJson.is_null()) 1511 { 1512 deleteIPAddress(ifaceId, nicIpEntry->id, asyncResp); 1513 } 1514 if (nicIpEntry != ipv6Data.cend()) 1515 { 1516 nicIpEntry = getNextStaticIpEntry(++nicIpEntry, 1517 ipv6Data.cend()); 1518 } 1519 entryIdx++; 1520 } 1521 } 1522 } 1523 1524 inline std::string extractParentInterfaceName(const std::string& ifaceId) 1525 { 1526 std::size_t pos = ifaceId.find('_'); 1527 return ifaceId.substr(0, pos); 1528 } 1529 1530 inline void 1531 parseInterfaceData(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1532 const std::string& ifaceId, 1533 const EthernetInterfaceData& ethData, 1534 const std::vector<IPv4AddressData>& ipv4Data, 1535 const std::vector<IPv6AddressData>& ipv6Data) 1536 { 1537 nlohmann::json& jsonResponse = asyncResp->res.jsonValue; 1538 jsonResponse["Id"] = ifaceId; 1539 jsonResponse["@odata.id"] = boost::urls::format( 1540 "/redfish/v1/Managers/bmc/EthernetInterfaces/{}", ifaceId); 1541 jsonResponse["InterfaceEnabled"] = ethData.nicEnabled; 1542 1543 if constexpr (bmcwebEnableHealthPopulate) 1544 { 1545 constexpr std::array<std::string_view, 1> inventoryForEthernet = { 1546 "xyz.openbmc_project.Inventory.Item.Ethernet"}; 1547 auto health = std::make_shared<HealthPopulate>(asyncResp); 1548 dbus::utility::getSubTreePaths( 1549 "/", 0, inventoryForEthernet, 1550 [health](const boost::system::error_code& ec, 1551 const dbus::utility::MapperGetSubTreePathsResponse& resp) { 1552 if (ec) 1553 { 1554 return; 1555 } 1556 1557 health->inventory = resp; 1558 }); 1559 1560 health->populate(); 1561 } 1562 1563 if (ethData.nicEnabled) 1564 { 1565 jsonResponse["LinkStatus"] = ethData.linkUp ? "LinkUp" : "LinkDown"; 1566 jsonResponse["Status"]["State"] = "Enabled"; 1567 } 1568 else 1569 { 1570 jsonResponse["LinkStatus"] = "NoLink"; 1571 jsonResponse["Status"]["State"] = "Disabled"; 1572 } 1573 1574 jsonResponse["SpeedMbps"] = ethData.speed; 1575 jsonResponse["MTUSize"] = ethData.mtuSize; 1576 jsonResponse["MACAddress"] = ethData.macAddress; 1577 jsonResponse["DHCPv4"]["DHCPEnabled"] = 1578 translateDhcpEnabledToBool(ethData.dhcpEnabled, true); 1579 jsonResponse["DHCPv4"]["UseNTPServers"] = ethData.ntpEnabled; 1580 jsonResponse["DHCPv4"]["UseDNSServers"] = ethData.dnsEnabled; 1581 jsonResponse["DHCPv4"]["UseDomainName"] = ethData.hostNameEnabled; 1582 1583 jsonResponse["DHCPv6"]["OperatingMode"] = 1584 translateDhcpEnabledToBool(ethData.dhcpEnabled, false) ? "Stateful" 1585 : "Disabled"; 1586 jsonResponse["DHCPv6"]["UseNTPServers"] = ethData.ntpEnabled; 1587 jsonResponse["DHCPv6"]["UseDNSServers"] = ethData.dnsEnabled; 1588 jsonResponse["DHCPv6"]["UseDomainName"] = ethData.hostNameEnabled; 1589 1590 if (!ethData.hostName.empty()) 1591 { 1592 jsonResponse["HostName"] = ethData.hostName; 1593 1594 // When domain name is empty then it means, that it is a network 1595 // without domain names, and the host name itself must be treated as 1596 // FQDN 1597 std::string fqdn = ethData.hostName; 1598 if (!ethData.domainnames.empty()) 1599 { 1600 fqdn += "." + ethData.domainnames[0]; 1601 } 1602 jsonResponse["FQDN"] = fqdn; 1603 } 1604 1605 if (ethData.vlanId) 1606 { 1607 jsonResponse["EthernetInterfaceType"] = "Virtual"; 1608 jsonResponse["VLAN"]["VLANEnable"] = true; 1609 jsonResponse["VLAN"]["VLANId"] = *ethData.vlanId; 1610 jsonResponse["VLAN"]["Tagged"] = true; 1611 1612 nlohmann::json::array_t relatedInterfaces; 1613 nlohmann::json& parentInterface = relatedInterfaces.emplace_back(); 1614 parentInterface["@odata.id"] = 1615 boost::urls::format("/redfish/v1/Managers/bmc/EthernetInterfaces", 1616 extractParentInterfaceName(ifaceId)); 1617 jsonResponse["Links"]["RelatedInterfaces"] = 1618 std::move(relatedInterfaces); 1619 } 1620 else 1621 { 1622 jsonResponse["EthernetInterfaceType"] = "Physical"; 1623 } 1624 1625 jsonResponse["NameServers"] = ethData.nameServers; 1626 jsonResponse["StaticNameServers"] = ethData.staticNameServers; 1627 1628 nlohmann::json& ipv4Array = jsonResponse["IPv4Addresses"]; 1629 nlohmann::json& ipv4StaticArray = jsonResponse["IPv4StaticAddresses"]; 1630 ipv4Array = nlohmann::json::array(); 1631 ipv4StaticArray = nlohmann::json::array(); 1632 for (const auto& ipv4Config : ipv4Data) 1633 { 1634 std::string gatewayStr = ipv4Config.gateway; 1635 if (gatewayStr.empty()) 1636 { 1637 gatewayStr = "0.0.0.0"; 1638 } 1639 nlohmann::json::object_t ipv4; 1640 ipv4["AddressOrigin"] = ipv4Config.origin; 1641 ipv4["SubnetMask"] = ipv4Config.netmask; 1642 ipv4["Address"] = ipv4Config.address; 1643 ipv4["Gateway"] = gatewayStr; 1644 1645 if (ipv4Config.origin == "Static") 1646 { 1647 ipv4StaticArray.push_back(ipv4); 1648 } 1649 1650 ipv4Array.emplace_back(std::move(ipv4)); 1651 } 1652 1653 std::string ipv6GatewayStr = ethData.ipv6DefaultGateway; 1654 if (ipv6GatewayStr.empty()) 1655 { 1656 ipv6GatewayStr = "0:0:0:0:0:0:0:0"; 1657 } 1658 1659 jsonResponse["IPv6DefaultGateway"] = ipv6GatewayStr; 1660 1661 nlohmann::json& ipv6Array = jsonResponse["IPv6Addresses"]; 1662 nlohmann::json& ipv6StaticArray = jsonResponse["IPv6StaticAddresses"]; 1663 ipv6Array = nlohmann::json::array(); 1664 ipv6StaticArray = nlohmann::json::array(); 1665 nlohmann::json& ipv6AddrPolicyTable = 1666 jsonResponse["IPv6AddressPolicyTable"]; 1667 ipv6AddrPolicyTable = nlohmann::json::array(); 1668 for (const auto& ipv6Config : ipv6Data) 1669 { 1670 nlohmann::json::object_t ipv6; 1671 ipv6["Address"] = ipv6Config.address; 1672 ipv6["PrefixLength"] = ipv6Config.prefixLength; 1673 ipv6["AddressOrigin"] = ipv6Config.origin; 1674 1675 ipv6Array.emplace_back(std::move(ipv6)); 1676 if (ipv6Config.origin == "Static") 1677 { 1678 nlohmann::json::object_t ipv6Static; 1679 ipv6Static["Address"] = ipv6Config.address; 1680 ipv6Static["PrefixLength"] = ipv6Config.prefixLength; 1681 ipv6StaticArray.emplace_back(std::move(ipv6Static)); 1682 } 1683 } 1684 } 1685 1686 inline void afterDelete(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1687 const std::string& ifaceId, 1688 const boost::system::error_code& ec, 1689 const sdbusplus::message_t& m) 1690 { 1691 if (!ec) 1692 { 1693 return; 1694 } 1695 const sd_bus_error* dbusError = m.get_error(); 1696 if (dbusError == nullptr) 1697 { 1698 messages::internalError(asyncResp->res); 1699 return; 1700 } 1701 BMCWEB_LOG_DEBUG << "DBus error: " << dbusError->name; 1702 1703 if (std::string_view("org.freedesktop.DBus.Error.UnknownObject") == 1704 dbusError->name) 1705 { 1706 messages::resourceNotFound(asyncResp->res, "EthernetInterface", 1707 ifaceId); 1708 return; 1709 } 1710 if (std::string_view("org.freedesktop.DBus.Error.UnknownMethod") == 1711 dbusError->name) 1712 { 1713 messages::resourceCannotBeDeleted(asyncResp->res); 1714 return; 1715 } 1716 messages::internalError(asyncResp->res); 1717 } 1718 1719 inline void afterVlanCreate(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1720 const std::string& parentInterfaceUri, 1721 const std::string& vlanInterface, 1722 const boost::system::error_code& ec, 1723 const sdbusplus::message_t& m 1724 1725 ) 1726 { 1727 if (ec) 1728 { 1729 const sd_bus_error* dbusError = m.get_error(); 1730 if (dbusError == nullptr) 1731 { 1732 messages::internalError(asyncResp->res); 1733 return; 1734 } 1735 BMCWEB_LOG_DEBUG << "DBus error: " << dbusError->name; 1736 1737 if (std::string_view( 1738 "xyz.openbmc_project.Common.Error.ResourceNotFound") == 1739 dbusError->name) 1740 { 1741 messages::propertyValueNotInList( 1742 asyncResp->res, parentInterfaceUri, 1743 "Links/RelatedInterfaces/0/@odata.id"); 1744 return; 1745 } 1746 if (std::string_view( 1747 "xyz.openbmc_project.Common.Error.InvalidArgument") == 1748 dbusError->name) 1749 { 1750 messages::resourceAlreadyExists(asyncResp->res, "EthernetInterface", 1751 "Id", vlanInterface); 1752 return; 1753 } 1754 messages::internalError(asyncResp->res); 1755 return; 1756 } 1757 1758 const boost::urls::url vlanInterfaceUri = boost::urls::format( 1759 "/redfish/v1/Managers/bmc/EthernetInterfaces/{}", vlanInterface); 1760 asyncResp->res.addHeader("Location", vlanInterfaceUri.buffer()); 1761 } 1762 1763 inline void requestEthernetInterfacesRoutes(App& app) 1764 { 1765 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/EthernetInterfaces/") 1766 .privileges(redfish::privileges::getEthernetInterfaceCollection) 1767 .methods(boost::beast::http::verb::get)( 1768 [&app](const crow::Request& req, 1769 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 1770 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 1771 { 1772 return; 1773 } 1774 1775 asyncResp->res.jsonValue["@odata.type"] = 1776 "#EthernetInterfaceCollection.EthernetInterfaceCollection"; 1777 asyncResp->res.jsonValue["@odata.id"] = 1778 "/redfish/v1/Managers/bmc/EthernetInterfaces"; 1779 asyncResp->res.jsonValue["Name"] = 1780 "Ethernet Network Interface Collection"; 1781 asyncResp->res.jsonValue["Description"] = 1782 "Collection of EthernetInterfaces for this Manager"; 1783 1784 // Get eth interface list, and call the below callback for JSON 1785 // preparation 1786 getEthernetIfaceList( 1787 [asyncResp](const bool& success, 1788 const std::vector<std::string>& ifaceList) { 1789 if (!success) 1790 { 1791 messages::internalError(asyncResp->res); 1792 return; 1793 } 1794 1795 nlohmann::json& ifaceArray = asyncResp->res.jsonValue["Members"]; 1796 ifaceArray = nlohmann::json::array(); 1797 for (const std::string& ifaceItem : ifaceList) 1798 { 1799 nlohmann::json::object_t iface; 1800 iface["@odata.id"] = boost::urls::format( 1801 "/redfish/v1/Managers/bmc/EthernetInterfaces/{}", 1802 ifaceItem); 1803 ifaceArray.push_back(std::move(iface)); 1804 } 1805 1806 asyncResp->res.jsonValue["Members@odata.count"] = ifaceArray.size(); 1807 asyncResp->res.jsonValue["@odata.id"] = 1808 "/redfish/v1/Managers/bmc/EthernetInterfaces"; 1809 }); 1810 }); 1811 1812 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/EthernetInterfaces/") 1813 .privileges(redfish::privileges::postEthernetInterfaceCollection) 1814 .methods(boost::beast::http::verb::post)( 1815 [&app](const crow::Request& req, 1816 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 1817 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 1818 { 1819 return; 1820 } 1821 1822 bool vlanEnable = false; 1823 uint32_t vlanId = 0; 1824 nlohmann::json::array_t relatedInterfaces; 1825 1826 if (!json_util::readJsonPatch(req, asyncResp->res, "VLAN/VLANEnable", 1827 vlanEnable, "VLAN/VLANId", vlanId, 1828 "Links/RelatedInterfaces", 1829 relatedInterfaces)) 1830 { 1831 return; 1832 } 1833 1834 if (relatedInterfaces.size() != 1) 1835 { 1836 messages::arraySizeTooLong(asyncResp->res, 1837 "Links/RelatedInterfaces", 1838 relatedInterfaces.size()); 1839 return; 1840 } 1841 1842 std::string parentInterfaceUri; 1843 if (!json_util::readJson(relatedInterfaces[0], asyncResp->res, 1844 "@odata.id", parentInterfaceUri)) 1845 { 1846 messages::propertyMissing(asyncResp->res, 1847 "Links/RelatedInterfaces/0/@odata.id"); 1848 return; 1849 } 1850 BMCWEB_LOG_INFO << "Parent Interface URI: " << parentInterfaceUri; 1851 1852 boost::urls::result<boost::urls::url_view> parsedUri = 1853 boost::urls::parse_relative_ref(parentInterfaceUri); 1854 if (!parsedUri) 1855 { 1856 messages::propertyValueFormatError( 1857 asyncResp->res, parentInterfaceUri, 1858 "Links/RelatedInterfaces/0/@odata.id"); 1859 return; 1860 } 1861 1862 std::string parentInterface; 1863 if (!crow::utility::readUrlSegments( 1864 *parsedUri, "redfish", "v1", "Managers", "bmc", 1865 "EthernetInterfaces", std::ref(parentInterface))) 1866 { 1867 messages::propertyValueNotInList( 1868 asyncResp->res, parentInterfaceUri, 1869 "Links/RelatedInterfaces/0/@odata.id"); 1870 return; 1871 } 1872 1873 if (!vlanEnable) 1874 { 1875 // In OpenBMC implementation, VLANEnable cannot be false on 1876 // create 1877 messages::propertyValueIncorrect(asyncResp->res, "VLAN/VLANEnable", 1878 "false"); 1879 return; 1880 } 1881 1882 std::string vlanInterface = parentInterface + "_" + 1883 std::to_string(vlanId); 1884 crow::connections::systemBus->async_method_call( 1885 [asyncResp, parentInterfaceUri, 1886 vlanInterface](const boost::system::error_code& ec, 1887 const sdbusplus::message_t& m) { 1888 afterVlanCreate(asyncResp, parentInterfaceUri, vlanInterface, ec, 1889 m); 1890 }, 1891 "xyz.openbmc_project.Network", "/xyz/openbmc_project/network", 1892 "xyz.openbmc_project.Network.VLAN.Create", "VLAN", parentInterface, 1893 vlanId); 1894 }); 1895 1896 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/EthernetInterfaces/<str>/") 1897 .privileges(redfish::privileges::getEthernetInterface) 1898 .methods(boost::beast::http::verb::get)( 1899 [&app](const crow::Request& req, 1900 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1901 const std::string& ifaceId) { 1902 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 1903 { 1904 return; 1905 } 1906 getEthernetIfaceData( 1907 ifaceId, 1908 [asyncResp, ifaceId](const bool& success, 1909 const EthernetInterfaceData& ethData, 1910 const std::vector<IPv4AddressData>& ipv4Data, 1911 const std::vector<IPv6AddressData>& ipv6Data) { 1912 if (!success) 1913 { 1914 // TODO(Pawel)consider distinguish between non 1915 // existing object, and other errors 1916 messages::resourceNotFound(asyncResp->res, "EthernetInterface", 1917 ifaceId); 1918 return; 1919 } 1920 1921 asyncResp->res.jsonValue["@odata.type"] = 1922 "#EthernetInterface.v1_9_0.EthernetInterface"; 1923 asyncResp->res.jsonValue["Name"] = "Manager Ethernet Interface"; 1924 asyncResp->res.jsonValue["Description"] = 1925 "Management Network Interface"; 1926 1927 parseInterfaceData(asyncResp, ifaceId, ethData, ipv4Data, ipv6Data); 1928 }); 1929 }); 1930 1931 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/EthernetInterfaces/<str>/") 1932 .privileges(redfish::privileges::patchEthernetInterface) 1933 .methods(boost::beast::http::verb::patch)( 1934 [&app](const crow::Request& req, 1935 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1936 const std::string& ifaceId) { 1937 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 1938 { 1939 return; 1940 } 1941 std::optional<std::string> hostname; 1942 std::optional<std::string> fqdn; 1943 std::optional<std::string> macAddress; 1944 std::optional<std::string> ipv6DefaultGateway; 1945 std::optional<nlohmann::json::array_t> ipv4StaticAddresses; 1946 std::optional<nlohmann::json::array_t> ipv6StaticAddresses; 1947 std::optional<std::vector<std::string>> staticNameServers; 1948 std::optional<nlohmann::json> dhcpv4; 1949 std::optional<nlohmann::json> dhcpv6; 1950 std::optional<bool> interfaceEnabled; 1951 std::optional<size_t> mtuSize; 1952 DHCPParameters v4dhcpParms; 1953 DHCPParameters v6dhcpParms; 1954 1955 if (!json_util::readJsonPatch( 1956 req, asyncResp->res, "HostName", hostname, "FQDN", fqdn, 1957 "IPv4StaticAddresses", ipv4StaticAddresses, "MACAddress", 1958 macAddress, "StaticNameServers", staticNameServers, 1959 "IPv6DefaultGateway", ipv6DefaultGateway, "IPv6StaticAddresses", 1960 ipv6StaticAddresses, "DHCPv4", dhcpv4, "DHCPv6", dhcpv6, 1961 "MTUSize", mtuSize, "InterfaceEnabled", interfaceEnabled)) 1962 { 1963 return; 1964 } 1965 if (dhcpv4) 1966 { 1967 if (!json_util::readJson(*dhcpv4, asyncResp->res, "DHCPEnabled", 1968 v4dhcpParms.dhcpv4Enabled, "UseDNSServers", 1969 v4dhcpParms.useDnsServers, "UseNTPServers", 1970 v4dhcpParms.useNtpServers, "UseDomainName", 1971 v4dhcpParms.useDomainName)) 1972 { 1973 return; 1974 } 1975 } 1976 1977 if (dhcpv6) 1978 { 1979 if (!json_util::readJson(*dhcpv6, asyncResp->res, "OperatingMode", 1980 v6dhcpParms.dhcpv6OperatingMode, 1981 "UseDNSServers", v6dhcpParms.useDnsServers, 1982 "UseNTPServers", v6dhcpParms.useNtpServers, 1983 "UseDomainName", 1984 v6dhcpParms.useDomainName)) 1985 { 1986 return; 1987 } 1988 } 1989 1990 // Get single eth interface data, and call the below callback 1991 // for JSON preparation 1992 getEthernetIfaceData( 1993 ifaceId, 1994 [asyncResp, ifaceId, hostname = std::move(hostname), 1995 fqdn = std::move(fqdn), macAddress = std::move(macAddress), 1996 ipv4StaticAddresses = std::move(ipv4StaticAddresses), 1997 ipv6DefaultGateway = std::move(ipv6DefaultGateway), 1998 ipv6StaticAddresses = std::move(ipv6StaticAddresses), 1999 staticNameServers = std::move(staticNameServers), 2000 dhcpv4 = std::move(dhcpv4), dhcpv6 = std::move(dhcpv6), mtuSize, 2001 v4dhcpParms = std::move(v4dhcpParms), 2002 v6dhcpParms = std::move(v6dhcpParms), interfaceEnabled]( 2003 const bool& success, const EthernetInterfaceData& ethData, 2004 const std::vector<IPv4AddressData>& ipv4Data, 2005 const std::vector<IPv6AddressData>& ipv6Data) { 2006 if (!success) 2007 { 2008 // ... otherwise return error 2009 // TODO(Pawel)consider distinguish between non 2010 // existing object, and other errors 2011 messages::resourceNotFound(asyncResp->res, "EthernetInterface", 2012 ifaceId); 2013 return; 2014 } 2015 2016 if (dhcpv4 || dhcpv6) 2017 { 2018 handleDHCPPatch(ifaceId, ethData, v4dhcpParms, v6dhcpParms, 2019 asyncResp); 2020 } 2021 2022 if (hostname) 2023 { 2024 handleHostnamePatch(*hostname, asyncResp); 2025 } 2026 2027 if (fqdn) 2028 { 2029 handleFqdnPatch(ifaceId, *fqdn, asyncResp); 2030 } 2031 2032 if (macAddress) 2033 { 2034 handleMACAddressPatch(ifaceId, *macAddress, asyncResp); 2035 } 2036 2037 if (ipv4StaticAddresses) 2038 { 2039 // TODO(ed) for some reason the capture of 2040 // ipv4Addresses above is returning a const value, 2041 // not a non-const value. This doesn't really work 2042 // for us, as we need to be able to efficiently move 2043 // out the intermedia nlohmann::json objects. This 2044 // makes a copy of the structure, and operates on 2045 // that, but could be done more efficiently 2046 nlohmann::json::array_t ipv4Static = *ipv4StaticAddresses; 2047 handleIPv4StaticPatch(ifaceId, ipv4Static, ipv4Data, asyncResp); 2048 } 2049 2050 if (staticNameServers) 2051 { 2052 handleStaticNameServersPatch(ifaceId, *staticNameServers, 2053 asyncResp); 2054 } 2055 2056 if (ipv6DefaultGateway) 2057 { 2058 messages::propertyNotWritable(asyncResp->res, 2059 "IPv6DefaultGateway"); 2060 } 2061 2062 if (ipv6StaticAddresses) 2063 { 2064 handleIPv6StaticAddressesPatch(ifaceId, *ipv6StaticAddresses, 2065 ipv6Data, asyncResp); 2066 } 2067 2068 if (interfaceEnabled) 2069 { 2070 setEthernetInterfaceBoolProperty(ifaceId, "NICEnabled", 2071 *interfaceEnabled, asyncResp); 2072 } 2073 2074 if (mtuSize) 2075 { 2076 handleMTUSizePatch(ifaceId, *mtuSize, asyncResp); 2077 } 2078 }); 2079 }); 2080 2081 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/EthernetInterfaces/<str>/") 2082 .privileges(redfish::privileges::deleteEthernetInterface) 2083 .methods(boost::beast::http::verb::delete_)( 2084 [&app](const crow::Request& req, 2085 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 2086 const std::string& ifaceId) { 2087 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 2088 { 2089 return; 2090 } 2091 2092 crow::connections::systemBus->async_method_call( 2093 [asyncResp, ifaceId](const boost::system::error_code& ec, 2094 const sdbusplus::message_t& m) { 2095 afterDelete(asyncResp, ifaceId, ec, m); 2096 }, 2097 "xyz.openbmc_project.Network", 2098 std::string("/xyz/openbmc_project/network/") + ifaceId, 2099 "xyz.openbmc_project.Object.Delete", "Delete"); 2100 }); 2101 } 2102 2103 } // namespace redfish 2104