blob: 533c7f5c2e42054a27c4c3e386af2cd3b309dcb7 [file] [log] [blame]
/*
// Copyright (c) 2018 Intel Corporation
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
*/
#pragma once
#include "app.hpp"
#include "dbus_singleton.hpp"
#include "dbus_utility.hpp"
#include "error_messages.hpp"
#include "human_sort.hpp"
#include "query.hpp"
#include "registries/privilege_registry.hpp"
#include "utils/ip_utils.hpp"
#include "utils/json_utils.hpp"
#include <boost/system/error_code.hpp>
#include <boost/url/format.hpp>
#include <array>
#include <cstddef>
#include <memory>
#include <optional>
#include <ranges>
#include <regex>
#include <string_view>
#include <variant>
#include <vector>
namespace redfish
{
enum class LinkType
{
Local,
Global
};
enum class IpVersion
{
IpV4,
IpV6
};
/**
* Structure for keeping IPv4 data required by Redfish
*/
struct IPv4AddressData
{
std::string id;
std::string address;
std::string domain;
std::string gateway;
std::string netmask;
std::string origin;
LinkType linktype{};
bool isActive{};
};
/**
* Structure for keeping IPv6 data required by Redfish
*/
struct IPv6AddressData
{
std::string id;
std::string address;
std::string origin;
uint8_t prefixLength = 0;
};
/**
* Structure for keeping static route data required by Redfish
*/
struct StaticGatewayData
{
std::string id;
std::string gateway;
size_t prefixLength = 0;
std::string protocol;
};
/**
* Structure for keeping basic single Ethernet Interface information
* available from DBus
*/
struct EthernetInterfaceData
{
uint32_t speed;
size_t mtuSize;
bool autoNeg;
bool dnsv4Enabled;
bool dnsv6Enabled;
bool domainv4Enabled;
bool domainv6Enabled;
bool ntpv4Enabled;
bool ntpv6Enabled;
bool hostNamev4Enabled;
bool hostNamev6Enabled;
bool linkUp;
bool nicEnabled;
bool ipv6AcceptRa;
std::string dhcpEnabled;
std::string operatingMode;
std::string hostName;
std::string defaultGateway;
std::string ipv6DefaultGateway;
std::string ipv6StaticDefaultGateway;
std::string macAddress;
std::optional<uint32_t> vlanId;
std::vector<std::string> nameServers;
std::vector<std::string> staticNameServers;
std::vector<std::string> domainnames;
};
struct DHCPParameters
{
std::optional<bool> dhcpv4Enabled;
std::optional<bool> useDnsServers;
std::optional<bool> useNtpServers;
std::optional<bool> useDomainName;
std::optional<std::string> dhcpv6OperatingMode;
};
// Helper function that changes bits netmask notation (i.e. /24)
// into full dot notation
inline std::string getNetmask(unsigned int bits)
{
uint32_t value = 0xffffffff << (32 - bits);
std::string netmask = std::to_string((value >> 24) & 0xff) + "." +
std::to_string((value >> 16) & 0xff) + "." +
std::to_string((value >> 8) & 0xff) + "." +
std::to_string(value & 0xff);
return netmask;
}
inline bool translateDhcpEnabledToBool(const std::string& inputDHCP,
bool isIPv4)
{
if (isIPv4)
{
return (
(inputDHCP ==
"xyz.openbmc_project.Network.EthernetInterface.DHCPConf.v4") ||
(inputDHCP ==
"xyz.openbmc_project.Network.EthernetInterface.DHCPConf.both"));
}
return ((inputDHCP ==
"xyz.openbmc_project.Network.EthernetInterface.DHCPConf.v6") ||
(inputDHCP ==
"xyz.openbmc_project.Network.EthernetInterface.DHCPConf.both"));
}
inline std::string getDhcpEnabledEnumeration(bool isIPv4, bool isIPv6)
{
if (isIPv4 && isIPv6)
{
return "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.both";
}
if (isIPv4)
{
return "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.v4";
}
if (isIPv6)
{
return "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.v6";
}
return "xyz.openbmc_project.Network.EthernetInterface.DHCPConf.none";
}
inline std::string
translateAddressOriginDbusToRedfish(const std::string& inputOrigin,
bool isIPv4)
{
if (inputOrigin == "xyz.openbmc_project.Network.IP.AddressOrigin.Static")
{
return "Static";
}
if (inputOrigin == "xyz.openbmc_project.Network.IP.AddressOrigin.LinkLocal")
{
if (isIPv4)
{
return "IPv4LinkLocal";
}
return "LinkLocal";
}
if (inputOrigin == "xyz.openbmc_project.Network.IP.AddressOrigin.DHCP")
{
if (isIPv4)
{
return "DHCP";
}
return "DHCPv6";
}
if (inputOrigin == "xyz.openbmc_project.Network.IP.AddressOrigin.SLAAC")
{
return "SLAAC";
}
return "";
}
inline bool extractEthernetInterfaceData(
const std::string& ethifaceId,
const dbus::utility::ManagedObjectType& dbusData,
EthernetInterfaceData& ethData)
{
bool idFound = false;
for (const auto& objpath : dbusData)
{
for (const auto& ifacePair : objpath.second)
{
if (objpath.first == "/xyz/openbmc_project/network/" + ethifaceId)
{
idFound = true;
if (ifacePair.first == "xyz.openbmc_project.Network.MACAddress")
{
for (const auto& propertyPair : ifacePair.second)
{
if (propertyPair.first == "MACAddress")
{
const std::string* mac =
std::get_if<std::string>(&propertyPair.second);
if (mac != nullptr)
{
ethData.macAddress = *mac;
}
}
}
}
else if (ifacePair.first == "xyz.openbmc_project.Network.VLAN")
{
for (const auto& propertyPair : ifacePair.second)
{
if (propertyPair.first == "Id")
{
const uint32_t* id =
std::get_if<uint32_t>(&propertyPair.second);
if (id != nullptr)
{
ethData.vlanId = *id;
}
}
}
}
else if (ifacePair.first ==
"xyz.openbmc_project.Network.EthernetInterface")
{
for (const auto& propertyPair : ifacePair.second)
{
if (propertyPair.first == "AutoNeg")
{
const bool* autoNeg =
std::get_if<bool>(&propertyPair.second);
if (autoNeg != nullptr)
{
ethData.autoNeg = *autoNeg;
}
}
else if (propertyPair.first == "Speed")
{
const uint32_t* speed =
std::get_if<uint32_t>(&propertyPair.second);
if (speed != nullptr)
{
ethData.speed = *speed;
}
}
else if (propertyPair.first == "MTU")
{
const size_t* mtuSize =
std::get_if<size_t>(&propertyPair.second);
if (mtuSize != nullptr)
{
ethData.mtuSize = *mtuSize;
}
}
else if (propertyPair.first == "LinkUp")
{
const bool* linkUp =
std::get_if<bool>(&propertyPair.second);
if (linkUp != nullptr)
{
ethData.linkUp = *linkUp;
}
}
else if (propertyPair.first == "NICEnabled")
{
const bool* nicEnabled =
std::get_if<bool>(&propertyPair.second);
if (nicEnabled != nullptr)
{
ethData.nicEnabled = *nicEnabled;
}
}
else if (propertyPair.first == "IPv6AcceptRA")
{
const bool* ipv6AcceptRa =
std::get_if<bool>(&propertyPair.second);
if (ipv6AcceptRa != nullptr)
{
ethData.ipv6AcceptRa = *ipv6AcceptRa;
}
}
else if (propertyPair.first == "Nameservers")
{
const std::vector<std::string>* nameservers =
std::get_if<std::vector<std::string>>(
&propertyPair.second);
if (nameservers != nullptr)
{
ethData.nameServers = *nameservers;
}
}
else if (propertyPair.first == "StaticNameServers")
{
const std::vector<std::string>* staticNameServers =
std::get_if<std::vector<std::string>>(
&propertyPair.second);
if (staticNameServers != nullptr)
{
ethData.staticNameServers = *staticNameServers;
}
}
else if (propertyPair.first == "DHCPEnabled")
{
const std::string* dhcpEnabled =
std::get_if<std::string>(&propertyPair.second);
if (dhcpEnabled != nullptr)
{
ethData.dhcpEnabled = *dhcpEnabled;
}
}
else if (propertyPair.first == "DomainName")
{
const std::vector<std::string>* domainNames =
std::get_if<std::vector<std::string>>(
&propertyPair.second);
if (domainNames != nullptr)
{
ethData.domainnames = *domainNames;
}
}
else if (propertyPair.first == "DefaultGateway")
{
const std::string* defaultGateway =
std::get_if<std::string>(&propertyPair.second);
if (defaultGateway != nullptr)
{
std::string defaultGatewayStr = *defaultGateway;
if (defaultGatewayStr.empty())
{
ethData.defaultGateway = "0.0.0.0";
}
else
{
ethData.defaultGateway = defaultGatewayStr;
}
}
}
else if (propertyPair.first == "DefaultGateway6")
{
const std::string* defaultGateway6 =
std::get_if<std::string>(&propertyPair.second);
if (defaultGateway6 != nullptr)
{
std::string defaultGateway6Str =
*defaultGateway6;
if (defaultGateway6Str.empty())
{
ethData.ipv6DefaultGateway =
"0:0:0:0:0:0:0:0";
}
else
{
ethData.ipv6DefaultGateway =
defaultGateway6Str;
}
}
}
}
}
}
sdbusplus::message::object_path path(
"/xyz/openbmc_project/network");
sdbusplus::message::object_path dhcp4Path = path / ethifaceId /
"dhcp4";
if (sdbusplus::message::object_path(objpath.first) == dhcp4Path)
{
if (ifacePair.first ==
"xyz.openbmc_project.Network.DHCPConfiguration")
{
for (const auto& propertyPair : ifacePair.second)
{
if (propertyPair.first == "DNSEnabled")
{
const bool* dnsEnabled =
std::get_if<bool>(&propertyPair.second);
if (dnsEnabled != nullptr)
{
ethData.dnsv4Enabled = *dnsEnabled;
}
}
else if (propertyPair.first == "DomainEnabled")
{
const bool* domainEnabled =
std::get_if<bool>(&propertyPair.second);
if (domainEnabled != nullptr)
{
ethData.domainv4Enabled = *domainEnabled;
}
}
else if (propertyPair.first == "NTPEnabled")
{
const bool* ntpEnabled =
std::get_if<bool>(&propertyPair.second);
if (ntpEnabled != nullptr)
{
ethData.ntpv4Enabled = *ntpEnabled;
}
}
else if (propertyPair.first == "HostNameEnabled")
{
const bool* hostNameEnabled =
std::get_if<bool>(&propertyPair.second);
if (hostNameEnabled != nullptr)
{
ethData.hostNamev4Enabled = *hostNameEnabled;
}
}
}
}
}
sdbusplus::message::object_path dhcp6Path = path / ethifaceId /
"dhcp6";
if (sdbusplus::message::object_path(objpath.first) == dhcp6Path)
{
if (ifacePair.first ==
"xyz.openbmc_project.Network.DHCPConfiguration")
{
for (const auto& propertyPair : ifacePair.second)
{
if (propertyPair.first == "DNSEnabled")
{
const bool* dnsEnabled =
std::get_if<bool>(&propertyPair.second);
if (dnsEnabled != nullptr)
{
ethData.dnsv6Enabled = *dnsEnabled;
}
}
if (propertyPair.first == "DomainEnabled")
{
const bool* domainEnabled =
std::get_if<bool>(&propertyPair.second);
if (domainEnabled != nullptr)
{
ethData.domainv6Enabled = *domainEnabled;
}
}
else if (propertyPair.first == "NTPEnabled")
{
const bool* ntpEnabled =
std::get_if<bool>(&propertyPair.second);
if (ntpEnabled != nullptr)
{
ethData.ntpv6Enabled = *ntpEnabled;
}
}
else if (propertyPair.first == "HostNameEnabled")
{
const bool* hostNameEnabled =
std::get_if<bool>(&propertyPair.second);
if (hostNameEnabled != nullptr)
{
ethData.hostNamev6Enabled = *hostNameEnabled;
}
}
}
}
}
// System configuration shows up in the global namespace, so no need
// to check eth number
if (ifacePair.first ==
"xyz.openbmc_project.Network.SystemConfiguration")
{
for (const auto& propertyPair : ifacePair.second)
{
if (propertyPair.first == "HostName")
{
const std::string* hostname =
std::get_if<std::string>(&propertyPair.second);
if (hostname != nullptr)
{
ethData.hostName = *hostname;
}
}
}
}
}
}
return idFound;
}
// Helper function that extracts data for single ethernet ipv6 address
inline void extractIPV6Data(const std::string& ethifaceId,
const dbus::utility::ManagedObjectType& dbusData,
std::vector<IPv6AddressData>& ipv6Config)
{
const std::string ipPathStart = "/xyz/openbmc_project/network/" +
ethifaceId;
// Since there might be several IPv6 configurations aligned with
// single ethernet interface, loop over all of them
for (const auto& objpath : dbusData)
{
// Check if proper pattern for object path appears
if (objpath.first.str.starts_with(ipPathStart + "/"))
{
for (const auto& interface : objpath.second)
{
if (interface.first == "xyz.openbmc_project.Network.IP")
{
auto type = std::ranges::find_if(interface.second,
[](const auto& property) {
return property.first == "Type";
});
if (type == interface.second.end())
{
continue;
}
const std::string* typeStr =
std::get_if<std::string>(&type->second);
if (typeStr == nullptr ||
(*typeStr !=
"xyz.openbmc_project.Network.IP.Protocol.IPv6"))
{
continue;
}
// Instance IPv6AddressData structure, and set as
// appropriate
IPv6AddressData& ipv6Address = ipv6Config.emplace_back();
ipv6Address.id =
objpath.first.str.substr(ipPathStart.size());
for (const auto& property : interface.second)
{
if (property.first == "Address")
{
const std::string* address =
std::get_if<std::string>(&property.second);
if (address != nullptr)
{
ipv6Address.address = *address;
}
}
else if (property.first == "Origin")
{
const std::string* origin =
std::get_if<std::string>(&property.second);
if (origin != nullptr)
{
ipv6Address.origin =
translateAddressOriginDbusToRedfish(*origin,
false);
}
}
else if (property.first == "PrefixLength")
{
const uint8_t* prefix =
std::get_if<uint8_t>(&property.second);
if (prefix != nullptr)
{
ipv6Address.prefixLength = *prefix;
}
}
else if (property.first == "Type" ||
property.first == "Gateway")
{
// Type & Gateway is not used
}
else
{
BMCWEB_LOG_ERROR(
"Got extra property: {} on the {} object",
property.first, objpath.first.str);
}
}
}
}
}
}
}
// Helper function that extracts data for single ethernet ipv4 address
inline void extractIPData(const std::string& ethifaceId,
const dbus::utility::ManagedObjectType& dbusData,
std::vector<IPv4AddressData>& ipv4Config)
{
const std::string ipPathStart = "/xyz/openbmc_project/network/" +
ethifaceId;
// Since there might be several IPv4 configurations aligned with
// single ethernet interface, loop over all of them
for (const auto& objpath : dbusData)
{
// Check if proper pattern for object path appears
if (objpath.first.str.starts_with(ipPathStart + "/"))
{
for (const auto& interface : objpath.second)
{
if (interface.first == "xyz.openbmc_project.Network.IP")
{
auto type = std::ranges::find_if(interface.second,
[](const auto& property) {
return property.first == "Type";
});
if (type == interface.second.end())
{
continue;
}
const std::string* typeStr =
std::get_if<std::string>(&type->second);
if (typeStr == nullptr ||
(*typeStr !=
"xyz.openbmc_project.Network.IP.Protocol.IPv4"))
{
continue;
}
// Instance IPv4AddressData structure, and set as
// appropriate
IPv4AddressData& ipv4Address = ipv4Config.emplace_back();
ipv4Address.id =
objpath.first.str.substr(ipPathStart.size());
for (const auto& property : interface.second)
{
if (property.first == "Address")
{
const std::string* address =
std::get_if<std::string>(&property.second);
if (address != nullptr)
{
ipv4Address.address = *address;
}
}
else if (property.first == "Origin")
{
const std::string* origin =
std::get_if<std::string>(&property.second);
if (origin != nullptr)
{
ipv4Address.origin =
translateAddressOriginDbusToRedfish(*origin,
true);
}
}
else if (property.first == "PrefixLength")
{
const uint8_t* mask =
std::get_if<uint8_t>(&property.second);
if (mask != nullptr)
{
// convert it to the string
ipv4Address.netmask = getNetmask(*mask);
}
}
else if (property.first == "Type" ||
property.first == "Gateway")
{
// Type & Gateway is not used
}
else
{
BMCWEB_LOG_ERROR(
"Got extra property: {} on the {} object",
property.first, objpath.first.str);
}
}
// Check if given address is local, or global
ipv4Address.linktype =
ipv4Address.address.starts_with("169.254.")
? LinkType::Local
: LinkType::Global;
}
}
}
}
}
/**
* @brief Modifies the default gateway assigned to the NIC
*
* @param[in] ifaceId Id of network interface whose default gateway is to be
* changed
* @param[in] gateway The new gateway value. Assigning an empty string
* causes the gateway to be deleted
* @param[io] asyncResp Response object that will be returned to client
*
* @return None
*/
inline void updateIPv4DefaultGateway(
const std::string& ifaceId, const std::string& gateway,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
setDbusProperty(
asyncResp, "xyz.openbmc_project.Network",
sdbusplus::message::object_path("/xyz/openbmc_project/network") /
ifaceId,
"xyz.openbmc_project.Network.EthernetInterface", "DefaultGateway",
"Gateway", gateway);
}
/**
* @brief Deletes given static IP address for the interface
*
* @param[in] ifaceId Id of interface whose IP should be deleted
* @param[in] ipHash DBus Hash id of IP that should be deleted
* @param[io] asyncResp Response object that will be returned to client
*
* @return None
*/
inline void deleteIPAddress(const std::string& ifaceId,
const std::string& ipHash,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
crow::connections::systemBus->async_method_call(
[asyncResp](const boost::system::error_code& ec) {
if (ec)
{
messages::internalError(asyncResp->res);
}
},
"xyz.openbmc_project.Network",
"/xyz/openbmc_project/network/" + ifaceId + ipHash,
"xyz.openbmc_project.Object.Delete", "Delete");
}
/**
* @brief Creates a static IPv4 entry
*
* @param[in] ifaceId Id of interface upon which to create the IPv4 entry
* @param[in] prefixLength IPv4 prefix syntax for the subnet mask
* @param[in] gateway IPv4 address of this interfaces gateway
* @param[in] address IPv4 address to assign to this interface
* @param[io] asyncResp Response object that will be returned to client
*
* @return None
*/
inline void createIPv4(const std::string& ifaceId, uint8_t prefixLength,
const std::string& gateway, const std::string& address,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
auto createIpHandler = [asyncResp, ifaceId,
gateway](const boost::system::error_code& ec) {
if (ec)
{
messages::internalError(asyncResp->res);
return;
}
};
crow::connections::systemBus->async_method_call(
std::move(createIpHandler), "xyz.openbmc_project.Network",
"/xyz/openbmc_project/network/" + ifaceId,
"xyz.openbmc_project.Network.IP.Create", "IP",
"xyz.openbmc_project.Network.IP.Protocol.IPv4", address, prefixLength,
gateway);
}
/**
* @brief Deletes the IP entry for this interface and creates a replacement
* static entry
*
* @param[in] ifaceId Id of interface upon which to create the IPv6 entry
* @param[in] id The unique hash entry identifying the DBus entry
* @param[in] prefixLength Prefix syntax for the subnet mask
* @param[in] address Address to assign to this interface
* @param[in] numStaticAddrs Count of IPv4 static addresses
* @param[io] asyncResp Response object that will be returned to client
*
* @return None
*/
inline void deleteAndCreateIPAddress(
IpVersion version, const std::string& ifaceId, const std::string& id,
uint8_t prefixLength, const std::string& address,
const std::string& gateway,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
crow::connections::systemBus->async_method_call(
[asyncResp, version, ifaceId, address, prefixLength,
gateway](const boost::system::error_code& ec) {
if (ec)
{
messages::internalError(asyncResp->res);
}
std::string protocol = "xyz.openbmc_project.Network.IP.Protocol.";
protocol += version == IpVersion::IpV4 ? "IPv4" : "IPv6";
crow::connections::systemBus->async_method_call(
[asyncResp](const boost::system::error_code& ec2) {
if (ec2)
{
messages::internalError(asyncResp->res);
}
},
"xyz.openbmc_project.Network",
"/xyz/openbmc_project/network/" + ifaceId,
"xyz.openbmc_project.Network.IP.Create", "IP", protocol, address,
prefixLength, gateway);
},
"xyz.openbmc_project.Network",
"/xyz/openbmc_project/network/" + ifaceId + id,
"xyz.openbmc_project.Object.Delete", "Delete");
}
inline bool extractIPv6DefaultGatewayData(
const std::string& ethifaceId,
const dbus::utility::ManagedObjectType& dbusData,
std::vector<StaticGatewayData>& staticGatewayConfig)
{
std::string staticGatewayPathStart("/xyz/openbmc_project/network/");
staticGatewayPathStart += ethifaceId;
for (const auto& objpath : dbusData)
{
if (!std::string_view(objpath.first.str)
.starts_with(staticGatewayPathStart))
{
continue;
}
for (const auto& interface : objpath.second)
{
if (interface.first != "xyz.openbmc_project.Network.StaticGateway")
{
continue;
}
StaticGatewayData& staticGateway =
staticGatewayConfig.emplace_back();
staticGateway.id = objpath.first.filename();
bool success = sdbusplus::unpackPropertiesNoThrow(
redfish::dbus_utils::UnpackErrorPrinter(), interface.second,
"Gateway", staticGateway.gateway, "PrefixLength",
staticGateway.prefixLength, "ProtocolType",
staticGateway.protocol);
if (!success)
{
return false;
}
}
}
return true;
}
/**
* @brief Creates IPv6 with given data
*
* @param[in] ifaceId Id of interface whose IP should be added
* @param[in] prefixLength Prefix length that needs to be added
* @param[in] address IP address that needs to be added
* @param[io] asyncResp Response object that will be returned to client
*
* @return None
*/
inline void createIPv6(const std::string& ifaceId, uint8_t prefixLength,
const std::string& address,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
sdbusplus::message::object_path path("/xyz/openbmc_project/network");
path /= ifaceId;
auto createIpHandler = [asyncResp,
address](const boost::system::error_code& ec) {
if (ec)
{
if (ec == boost::system::errc::io_error)
{
messages::propertyValueFormatError(asyncResp->res, address,
"Address");
}
else
{
messages::internalError(asyncResp->res);
}
}
};
// Passing null for gateway, as per redfish spec IPv6StaticAddresses
// object does not have associated gateway property
crow::connections::systemBus->async_method_call(
std::move(createIpHandler), "xyz.openbmc_project.Network", path,
"xyz.openbmc_project.Network.IP.Create", "IP",
"xyz.openbmc_project.Network.IP.Protocol.IPv6", address, prefixLength,
"");
}
/**
* @brief Deletes given IPv6 Static Gateway
*
* @param[in] ifaceId Id of interface whose IP should be deleted
* @param[in] ipHash DBus Hash id of IP that should be deleted
* @param[io] asyncResp Response object that will be returned to client
*
* @return None
*/
inline void
deleteIPv6Gateway(std::string_view gatewayId,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
sdbusplus::message::object_path path("/xyz/openbmc_project/network");
path /= gatewayId;
crow::connections::systemBus->async_method_call(
[asyncResp](const boost::system::error_code& ec) {
if (ec)
{
messages::internalError(asyncResp->res);
}
},
"xyz.openbmc_project.Network", path,
"xyz.openbmc_project.Object.Delete", "Delete");
}
/**
* @brief Creates IPv6 static default gateway with given data
*
* @param[in] ifaceId Id of interface whose IP should be added
* @param[in] prefixLength Prefix length that needs to be added
* @param[in] gateway Gateway address that needs to be added
* @param[io] asyncResp Response object that will be returned to client
*
* @return None
*/
inline void createIPv6DefaultGateway(
std::string_view ifaceId, size_t prefixLength, std::string_view gateway,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
sdbusplus::message::object_path path("/xyz/openbmc_project/network");
path /= ifaceId;
auto createIpHandler = [asyncResp](const boost::system::error_code& ec) {
if (ec)
{
messages::internalError(asyncResp->res);
}
};
crow::connections::systemBus->async_method_call(
std::move(createIpHandler), "xyz.openbmc_project.Network", path,
"xyz.openbmc_project.Network.StaticGateway.Create", "StaticGateway",
gateway, prefixLength, "xyz.openbmc_project.Network.IP.Protocol.IPv6");
}
/**
* @brief Deletes the IPv6 default gateway entry for this interface and
* creates a replacement IPv6 default gateway entry
*
* @param[in] ifaceId Id of interface upon which to create the IPv6
* entry
* @param[in] gateway IPv6 gateway to assign to this interface
* @param[in] prefixLength IPv6 prefix syntax for the subnet mask
* @param[io] asyncResp Response object that will be returned to client
*
* @return None
*/
inline void deleteAndCreateIPv6DefaultGateway(
std::string_view ifaceId, std::string_view gatewayId,
std::string_view gateway, size_t prefixLength,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
sdbusplus::message::object_path path("/xyz/openbmc_project/network");
path /= gatewayId;
crow::connections::systemBus->async_method_call(
[asyncResp, ifaceId, gateway,
prefixLength](const boost::system::error_code& ec) {
if (ec)
{
messages::internalError(asyncResp->res);
return;
}
createIPv6DefaultGateway(ifaceId, prefixLength, gateway, asyncResp);
},
"xyz.openbmc_project.Network", path,
"xyz.openbmc_project.Object.Delete", "Delete");
}
/**
* @brief Sets IPv6 default gateway with given data
*
* @param[in] ifaceId Id of interface whose gateway should be added
* @param[in] input Contains address that needs to be added
* @param[in] staticGatewayData Current static gateways in the system
* @param[io] asyncResp Response object that will be returned to client
*
* @return None
*/
inline void handleIPv6DefaultGateway(
const std::string& ifaceId,
std::vector<std::variant<nlohmann::json::object_t, std::nullptr_t>>& input,
const std::vector<StaticGatewayData>& staticGatewayData,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
size_t entryIdx = 1;
std::vector<StaticGatewayData>::const_iterator staticGatewayEntry =
staticGatewayData.begin();
for (std::variant<nlohmann::json::object_t, std::nullptr_t>& thisJson :
input)
{
// find the next gateway entry
while (staticGatewayEntry != staticGatewayData.end())
{
if (staticGatewayEntry->protocol ==
"xyz.openbmc_project.Network.IP.Protocol.IPv6")
{
break;
}
staticGatewayEntry++;
}
std::string pathString = "IPv6StaticDefaultGateways/" +
std::to_string(entryIdx);
nlohmann::json::object_t* obj =
std::get_if<nlohmann::json::object_t>(&thisJson);
if (obj == nullptr)
{
if (staticGatewayEntry == staticGatewayData.end())
{
messages::resourceCannotBeDeleted(asyncResp->res);
return;
}
deleteIPv6Gateway(staticGatewayEntry->id, asyncResp);
return;
}
if (obj->empty())
{
// Do nothing, but make sure the entry exists.
if (staticGatewayEntry == staticGatewayData.end())
{
messages::propertyValueFormatError(asyncResp->res, *obj,
pathString);
return;
}
}
std::optional<std::string> address;
std::optional<size_t> prefixLength;
if (!json_util::readJsonObject(*obj, asyncResp->res, "Address", address,
"PrefixLength", prefixLength))
{
return;
}
const std::string* addr = nullptr;
size_t prefix = 0;
if (address)
{
addr = &(*address);
}
else if (staticGatewayEntry != staticGatewayData.end())
{
addr = &(staticGatewayEntry->gateway);
}
else
{
messages::propertyMissing(asyncResp->res, pathString + "/Address");
return;
}
if (prefixLength)
{
prefix = *prefixLength;
}
else if (staticGatewayEntry != staticGatewayData.end())
{
prefix = staticGatewayEntry->prefixLength;
}
else
{
messages::propertyMissing(asyncResp->res,
pathString + "/PrefixLength");
return;
}
if (staticGatewayEntry != staticGatewayData.end())
{
deleteAndCreateIPv6DefaultGateway(ifaceId, staticGatewayEntry->id,
*addr, prefix, asyncResp);
staticGatewayEntry++;
}
else
{
createIPv6DefaultGateway(ifaceId, prefix, *addr, asyncResp);
}
entryIdx++;
}
}
/**
* Function that retrieves all properties for given Ethernet Interface
* Object
* from EntityManager Network Manager
* @param ethiface_id a eth interface id to query on DBus
* @param callback a function that shall be called to convert Dbus output
* into JSON
*/
template <typename CallbackFunc>
void getEthernetIfaceData(const std::string& ethifaceId,
CallbackFunc&& callback)
{
sdbusplus::message::object_path path("/xyz/openbmc_project/network");
dbus::utility::getManagedObjects(
"xyz.openbmc_project.Network", path,
[ethifaceId{std::string{ethifaceId}},
callback = std::forward<CallbackFunc>(callback)](
const boost::system::error_code& ec,
const dbus::utility::ManagedObjectType& resp) mutable {
EthernetInterfaceData ethData{};
std::vector<IPv4AddressData> ipv4Data;
std::vector<IPv6AddressData> ipv6Data;
std::vector<StaticGatewayData> ipv6GatewayData;
if (ec)
{
callback(false, ethData, ipv4Data, ipv6Data, ipv6GatewayData);
return;
}
bool found = extractEthernetInterfaceData(ethifaceId, resp, ethData);
if (!found)
{
callback(false, ethData, ipv4Data, ipv6Data, ipv6GatewayData);
return;
}
extractIPData(ethifaceId, resp, ipv4Data);
// Fix global GW
for (IPv4AddressData& ipv4 : ipv4Data)
{
if (((ipv4.linktype == LinkType::Global) &&
(ipv4.gateway == "0.0.0.0")) ||
(ipv4.origin == "DHCP") || (ipv4.origin == "Static"))
{
ipv4.gateway = ethData.defaultGateway;
}
}
extractIPV6Data(ethifaceId, resp, ipv6Data);
if (!extractIPv6DefaultGatewayData(ethifaceId, resp, ipv6GatewayData))
{
callback(false, ethData, ipv4Data, ipv6Data, ipv6GatewayData);
}
// Finally make a callback with useful data
callback(true, ethData, ipv4Data, ipv6Data, ipv6GatewayData);
});
}
/**
* Function that retrieves all Ethernet Interfaces available through Network
* Manager
* @param callback a function that shall be called to convert Dbus output
* into JSON.
*/
template <typename CallbackFunc>
void getEthernetIfaceList(CallbackFunc&& callback)
{
sdbusplus::message::object_path path("/xyz/openbmc_project/network");
dbus::utility::getManagedObjects(
"xyz.openbmc_project.Network", path,
[callback = std::forward<CallbackFunc>(callback)](
const boost::system::error_code& ec,
const dbus::utility::ManagedObjectType& resp) {
// Callback requires vector<string> to retrieve all available
// ethernet interfaces
std::vector<std::string> ifaceList;
ifaceList.reserve(resp.size());
if (ec)
{
callback(false, ifaceList);
return;
}
// Iterate over all retrieved ObjectPaths.
for (const auto& objpath : resp)
{
// And all interfaces available for certain ObjectPath.
for (const auto& interface : objpath.second)
{
// If interface is
// xyz.openbmc_project.Network.EthernetInterface, this is
// what we're looking for.
if (interface.first ==
"xyz.openbmc_project.Network.EthernetInterface")
{
std::string ifaceId = objpath.first.filename();
if (ifaceId.empty())
{
continue;
}
// and put it into output vector.
ifaceList.emplace_back(ifaceId);
}
}
}
std::ranges::sort(ifaceList, AlphanumLess<std::string>());
// Finally make a callback with useful data
callback(true, ifaceList);
});
}
inline void
handleHostnamePatch(const std::string& hostname,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
// SHOULD handle host names of up to 255 characters(RFC 1123)
if (hostname.length() > 255)
{
messages::propertyValueFormatError(asyncResp->res, hostname,
"HostName");
return;
}
setDbusProperty(
asyncResp, "xyz.openbmc_project.Network",
sdbusplus::message::object_path("/xyz/openbmc_project/network/config"),
"xyz.openbmc_project.Network.SystemConfiguration", "HostName",
"HostName", hostname);
}
inline void
handleMTUSizePatch(const std::string& ifaceId, const size_t mtuSize,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
sdbusplus::message::object_path objPath("/xyz/openbmc_project/network");
objPath /= ifaceId;
setDbusProperty(asyncResp, "xyz.openbmc_project.Network", objPath,
"xyz.openbmc_project.Network.EthernetInterface", "MTU",
"MTUSize", mtuSize);
}
inline void
handleDomainnamePatch(const std::string& ifaceId,
const std::string& domainname,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
std::vector<std::string> vectorDomainname = {domainname};
setDbusProperty(
asyncResp, "xyz.openbmc_project.Network",
sdbusplus::message::object_path("/xyz/openbmc_project/network") /
ifaceId,
"xyz.openbmc_project.Network.EthernetInterface", "DomainName", "FQDN",
vectorDomainname);
}
inline bool isHostnameValid(const std::string& hostname)
{
// A valid host name can never have the dotted-decimal form (RFC 1123)
if (std::ranges::all_of(hostname, ::isdigit))
{
return false;
}
// Each label(hostname/subdomains) within a valid FQDN
// MUST handle host names of up to 63 characters (RFC 1123)
// labels cannot start or end with hyphens (RFC 952)
// labels can start with numbers (RFC 1123)
const static std::regex pattern(
"^[a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\\-]{0,61}[a-zA-Z0-9]$");
return std::regex_match(hostname, pattern);
}
inline bool isDomainnameValid(const std::string& domainname)
{
// Can have multiple subdomains
// Top Level Domain's min length is 2 character
const static std::regex pattern(
"^([A-Za-z0-9][a-zA-Z0-9\\-]{1,61}|[a-zA-Z0-9]{1,30}\\.)*[a-zA-Z]{2,}$");
return std::regex_match(domainname, pattern);
}
inline void handleFqdnPatch(const std::string& ifaceId, const std::string& fqdn,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
// Total length of FQDN must not exceed 255 characters(RFC 1035)
if (fqdn.length() > 255)
{
messages::propertyValueFormatError(asyncResp->res, fqdn, "FQDN");
return;
}
size_t pos = fqdn.find('.');
if (pos == std::string::npos)
{
messages::propertyValueFormatError(asyncResp->res, fqdn, "FQDN");
return;
}
std::string hostname;
std::string domainname;
domainname = (fqdn).substr(pos + 1);
hostname = (fqdn).substr(0, pos);
if (!isHostnameValid(hostname) || !isDomainnameValid(domainname))
{
messages::propertyValueFormatError(asyncResp->res, fqdn, "FQDN");
return;
}
handleHostnamePatch(hostname, asyncResp);
handleDomainnamePatch(ifaceId, domainname, asyncResp);
}
inline void
handleMACAddressPatch(const std::string& ifaceId,
const std::string& macAddress,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
setDbusProperty(
asyncResp, "xyz.openbmc_project.Network",
sdbusplus::message::object_path("/xyz/openbmc_project/network") /
ifaceId,
"xyz.openbmc_project.Network.MACAddress", "MACAddress", "MACAddress",
macAddress);
}
inline void setDHCPEnabled(const std::string& ifaceId,
const std::string& propertyName, const bool v4Value,
const bool v6Value,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
const std::string dhcp = getDhcpEnabledEnumeration(v4Value, v6Value);
setDbusProperty(
asyncResp, "xyz.openbmc_project.Network",
sdbusplus::message::object_path("/xyz/openbmc_project/network") /
ifaceId,
"xyz.openbmc_project.Network.EthernetInterface", propertyName, "DHCPv4",
dhcp);
}
enum class NetworkType
{
dhcp4,
dhcp6
};
inline void setDHCPConfig(const std::string& propertyName, const bool& value,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
const std::string& ethifaceId, NetworkType type)
{
BMCWEB_LOG_DEBUG("{} = {}", propertyName, value);
std::string redfishPropertyName;
sdbusplus::message::object_path path("/xyz/openbmc_project/network/");
path /= ethifaceId;
if (type == NetworkType::dhcp4)
{
path /= "dhcp4";
redfishPropertyName = "DHCPv4";
}
else
{
path /= "dhcp6";
redfishPropertyName = "DHCPv6";
}
setDbusProperty(asyncResp, "xyz.openbmc_project.Network", path,
"xyz.openbmc_project.Network.DHCPConfiguration",
propertyName, redfishPropertyName, value);
}
inline void handleSLAACAutoConfigPatch(
const std::string& ifaceId, bool ipv6AutoConfigEnabled,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
sdbusplus::message::object_path path("/xyz/openbmc_project/network");
path /= ifaceId;
setDbusProperty(asyncResp, "xyz.openbmc_project.Network", path,
"xyz.openbmc_project.Network.EthernetInterface",
"IPv6AcceptRA",
"StatelessAddressAutoConfig/IPv6AutoConfigEnabled",
ipv6AutoConfigEnabled);
}
inline void handleDHCPPatch(const std::string& ifaceId,
const EthernetInterfaceData& ethData,
const DHCPParameters& v4dhcpParms,
const DHCPParameters& v6dhcpParms,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
bool ipv4Active = translateDhcpEnabledToBool(ethData.dhcpEnabled, true);
bool ipv6Active = translateDhcpEnabledToBool(ethData.dhcpEnabled, false);
if (ipv4Active)
{
updateIPv4DefaultGateway(ifaceId, "", asyncResp);
}
bool nextv4DHCPState =
v4dhcpParms.dhcpv4Enabled ? *v4dhcpParms.dhcpv4Enabled : ipv4Active;
bool nextv6DHCPState{};
if (v6dhcpParms.dhcpv6OperatingMode)
{
if ((*v6dhcpParms.dhcpv6OperatingMode != "Enabled") &&
(*v6dhcpParms.dhcpv6OperatingMode != "Disabled"))
{
messages::propertyValueFormatError(asyncResp->res,
*v6dhcpParms.dhcpv6OperatingMode,
"OperatingMode");
return;
}
nextv6DHCPState = (*v6dhcpParms.dhcpv6OperatingMode == "Enabled");
}
else
{
nextv6DHCPState = ipv6Active;
}
bool nextDNSv4 = ethData.dnsv4Enabled;
bool nextDNSv6 = ethData.dnsv6Enabled;
if (v4dhcpParms.useDnsServers)
{
nextDNSv4 = *v4dhcpParms.useDnsServers;
}
if (v6dhcpParms.useDnsServers)
{
nextDNSv6 = *v6dhcpParms.useDnsServers;
}
bool nextNTPv4 = ethData.ntpv4Enabled;
bool nextNTPv6 = ethData.ntpv6Enabled;
if (v4dhcpParms.useNtpServers)
{
nextNTPv4 = *v4dhcpParms.useNtpServers;
}
if (v6dhcpParms.useNtpServers)
{
nextNTPv6 = *v6dhcpParms.useNtpServers;
}
bool nextUsev4Domain = ethData.domainv4Enabled;
bool nextUsev6Domain = ethData.domainv6Enabled;
if (v4dhcpParms.useDomainName)
{
nextUsev4Domain = *v4dhcpParms.useDomainName;
}
if (v6dhcpParms.useDomainName)
{
nextUsev6Domain = *v6dhcpParms.useDomainName;
}
BMCWEB_LOG_DEBUG("set DHCPEnabled...");
setDHCPEnabled(ifaceId, "DHCPEnabled", nextv4DHCPState, nextv6DHCPState,
asyncResp);
BMCWEB_LOG_DEBUG("set DNSEnabled...");
setDHCPConfig("DNSEnabled", nextDNSv4, asyncResp, ifaceId,
NetworkType::dhcp4);
BMCWEB_LOG_DEBUG("set NTPEnabled...");
setDHCPConfig("NTPEnabled", nextNTPv4, asyncResp, ifaceId,
NetworkType::dhcp4);
BMCWEB_LOG_DEBUG("set DomainEnabled...");
setDHCPConfig("DomainEnabled", nextUsev4Domain, asyncResp, ifaceId,
NetworkType::dhcp4);
BMCWEB_LOG_DEBUG("set DNSEnabled for dhcp6...");
setDHCPConfig("DNSEnabled", nextDNSv6, asyncResp, ifaceId,
NetworkType::dhcp6);
BMCWEB_LOG_DEBUG("set NTPEnabled for dhcp6...");
setDHCPConfig("NTPEnabled", nextNTPv6, asyncResp, ifaceId,
NetworkType::dhcp6);
BMCWEB_LOG_DEBUG("set DomainEnabled for dhcp6...");
setDHCPConfig("DomainEnabled", nextUsev6Domain, asyncResp, ifaceId,
NetworkType::dhcp6);
}
inline std::vector<IPv4AddressData>::const_iterator getNextStaticIpEntry(
const std::vector<IPv4AddressData>::const_iterator& head,
const std::vector<IPv4AddressData>::const_iterator& end)
{
return std::find_if(head, end, [](const IPv4AddressData& value) {
return value.origin == "Static";
});
}
inline std::vector<IPv6AddressData>::const_iterator getNextStaticIpEntry(
const std::vector<IPv6AddressData>::const_iterator& head,
const std::vector<IPv6AddressData>::const_iterator& end)
{
return std::find_if(head, end, [](const IPv6AddressData& value) {
return value.origin == "Static";
});
}
inline void handleIPv4StaticPatch(
const std::string& ifaceId,
std::vector<std::variant<nlohmann::json::object_t, std::nullptr_t>>& input,
const EthernetInterfaceData& ethData,
const std::vector<IPv4AddressData>& ipv4Data,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
unsigned entryIdx = 1;
// Find the first static IP address currently active on the NIC and
// match it to the first JSON element in the IPv4StaticAddresses array.
// Match each subsequent JSON element to the next static IP programmed
// into the NIC.
std::vector<IPv4AddressData>::const_iterator nicIpEntry =
getNextStaticIpEntry(ipv4Data.cbegin(), ipv4Data.cend());
bool gatewayValueAssigned{};
std::string activePath{};
std::string activeGateway{};
if (!ethData.defaultGateway.empty() && ethData.defaultGateway != "0.0.0.0")
{
// The NIC is already configured with a default gateway. Use this if
// the leading entry in the PATCH is '{}', which is preserving an active
// static address.
activeGateway = ethData.defaultGateway;
activePath = "IPv4StaticAddresses/1";
gatewayValueAssigned = true;
}
for (std::variant<nlohmann::json::object_t, std::nullptr_t>& thisJson :
input)
{
std::string pathString = "IPv4StaticAddresses/" +
std::to_string(entryIdx);
nlohmann::json::object_t* obj =
std::get_if<nlohmann::json::object_t>(&thisJson);
if (obj == nullptr)
{
if (nicIpEntry != ipv4Data.cend())
{
deleteIPAddress(ifaceId, nicIpEntry->id, asyncResp);
nicIpEntry = getNextStaticIpEntry(++nicIpEntry,
ipv4Data.cend());
if (!gatewayValueAssigned && (nicIpEntry == ipv4Data.cend()))
{
// All entries have been processed, and this last has
// requested the IP address be deleted. No prior entry
// performed an action that created or modified a
// gateway. Deleting this IP address means the default
// gateway entry has to be removed as well.
updateIPv4DefaultGateway(ifaceId, "", asyncResp);
}
entryIdx++;
continue;
}
// Received a DELETE action on an entry not assigned to the NIC
messages::resourceCannotBeDeleted(asyncResp->res);
return;
}
// An Add/Modify action is requested
if (!obj->empty())
{
std::optional<std::string> address;
std::optional<std::string> subnetMask;
std::optional<std::string> gateway;
if (!json_util::readJsonObject(*obj, asyncResp->res, "Address",
address, "SubnetMask", subnetMask,
"Gateway", gateway))
{
messages::propertyValueFormatError(asyncResp->res, *obj,
pathString);
return;
}
// Find the address/subnet/gateway values. Any values that are
// not explicitly provided are assumed to be unmodified from the
// current state of the interface. Merge existing state into the
// current request.
if (address)
{
if (!ip_util::ipv4VerifyIpAndGetBitcount(*address))
{
messages::propertyValueFormatError(asyncResp->res, *address,
pathString + "/Address");
return;
}
}
else if (nicIpEntry != ipv4Data.cend())
{
address = (nicIpEntry->address);
}
else
{
messages::propertyMissing(asyncResp->res,
pathString + "/Address");
return;
}
uint8_t prefixLength = 0;
if (subnetMask)
{
if (!ip_util::ipv4VerifyIpAndGetBitcount(*subnetMask,
&prefixLength))
{
messages::propertyValueFormatError(
asyncResp->res, *subnetMask,
pathString + "/SubnetMask");
return;
}
}
else if (nicIpEntry != ipv4Data.cend())
{
if (!ip_util::ipv4VerifyIpAndGetBitcount(nicIpEntry->netmask,
&prefixLength))
{
messages::propertyValueFormatError(
asyncResp->res, nicIpEntry->netmask,
pathString + "/SubnetMask");
return;
}
}
else
{
messages::propertyMissing(asyncResp->res,
pathString + "/SubnetMask");
return;
}
if (gateway)
{
if (!ip_util::ipv4VerifyIpAndGetBitcount(*gateway))
{
messages::propertyValueFormatError(asyncResp->res, *gateway,
pathString + "/Gateway");
return;
}
}
else if (nicIpEntry != ipv4Data.cend())
{
gateway = nicIpEntry->gateway;
}
else
{
messages::propertyMissing(asyncResp->res,
pathString + "/Gateway");
return;
}
if (gatewayValueAssigned)
{
if (activeGateway != gateway)
{
// A NIC can only have a single active gateway value.
// If any gateway in the array of static addresses
// mismatch the PATCH is in error.
std::string arg1 = pathString + "/Gateway";
std::string arg2 = activePath + "/Gateway";
messages::propertyValueConflict(asyncResp->res, arg1, arg2);
return;
}
}
else
{
// Capture the very first gateway value from the incoming
// JSON record and use it at the default gateway.
updateIPv4DefaultGateway(ifaceId, *gateway, asyncResp);
activeGateway = *gateway;
activePath = pathString;
gatewayValueAssigned = true;
}
if (nicIpEntry != ipv4Data.cend())
{
deleteAndCreateIPAddress(IpVersion::IpV4, ifaceId,
nicIpEntry->id, prefixLength, *address,
*gateway, asyncResp);
nicIpEntry = getNextStaticIpEntry(++nicIpEntry,
ipv4Data.cend());
}
else
{
createIPv4(ifaceId, prefixLength, *gateway, *address,
asyncResp);
}
entryIdx++;
}
else
{
// Received {}, do not modify this address
if (nicIpEntry != ipv4Data.cend())
{
nicIpEntry = getNextStaticIpEntry(++nicIpEntry,
ipv4Data.cend());
entryIdx++;
}
else
{
// Requested a DO NOT MODIFY action on an entry not assigned
// to the NIC
messages::propertyValueFormatError(asyncResp->res, *obj,
pathString);
return;
}
}
}
}
inline void handleStaticNameServersPatch(
const std::string& ifaceId,
const std::vector<std::string>& updatedStaticNameServers,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
setDbusProperty(
asyncResp, "xyz.openbmc_project.Network",
sdbusplus::message::object_path("/xyz/openbmc_project/network") /
ifaceId,
"xyz.openbmc_project.Network.EthernetInterface", "StaticNameServers",
"StaticNameServers", updatedStaticNameServers);
}
inline void handleIPv6StaticAddressesPatch(
const std::string& ifaceId,
std::vector<std::variant<nlohmann::json::object_t, std::nullptr_t>>& input,
const std::vector<IPv6AddressData>& ipv6Data,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp)
{
size_t entryIdx = 1;
std::vector<IPv6AddressData>::const_iterator nicIpEntry =
getNextStaticIpEntry(ipv6Data.cbegin(), ipv6Data.cend());
for (std::variant<nlohmann::json::object_t, std::nullptr_t>& thisJson :
input)
{
std::string pathString = "IPv6StaticAddresses/" +
std::to_string(entryIdx);
nlohmann::json::object_t* obj =
std::get_if<nlohmann::json::object_t>(&thisJson);
if (obj != nullptr && !obj->empty())
{
std::optional<std::string> address;
std::optional<uint8_t> prefixLength;
nlohmann::json::object_t thisJsonCopy = *obj;
if (!json_util::readJsonObject(thisJsonCopy, asyncResp->res,
"Address", address, "PrefixLength",
prefixLength))
{
messages::propertyValueFormatError(asyncResp->res, thisJsonCopy,
pathString);
return;
}
// Find the address and prefixLength values. Any values that are
// not explicitly provided are assumed to be unmodified from the
// current state of the interface. Merge existing state into the
// current request.
if (!address)
{
if (nicIpEntry == ipv6Data.end())
{
messages::propertyMissing(asyncResp->res,
pathString + "/Address");
return;
}
address = nicIpEntry->address;
}
if (!prefixLength)
{
if (nicIpEntry == ipv6Data.end())
{
messages::propertyMissing(asyncResp->res,
pathString + "/PrefixLength");
return;
}
prefixLength = nicIpEntry->prefixLength;
}
if (nicIpEntry != ipv6Data.end())
{
deleteAndCreateIPAddress(IpVersion::IpV6, ifaceId,
nicIpEntry->id, *prefixLength,
*address, "", asyncResp);
nicIpEntry = getNextStaticIpEntry(++nicIpEntry,
ipv6Data.cend());
}
else
{
createIPv6(ifaceId, *prefixLength, *address, asyncResp);
}
entryIdx++;
}
else
{
if (nicIpEntry == ipv6Data.end())
{
// Requesting a DELETE/DO NOT MODIFY action for an item
// that isn't present on the eth(n) interface. Input JSON is
// in error, so bail out.
if (obj == nullptr)
{
messages::resourceCannotBeDeleted(asyncResp->res);
return;
}
messages::propertyValueFormatError(asyncResp->res, *obj,
pathString);
return;
}
if (obj == nullptr)
{
deleteIPAddress(ifaceId, nicIpEntry->id, asyncResp);
}
if (nicIpEntry != ipv6Data.cend())
{
nicIpEntry = getNextStaticIpEntry(++nicIpEntry,
ipv6Data.cend());
}
entryIdx++;
}
}
}
inline std::string extractParentInterfaceName(const std::string& ifaceId)
{
std::size_t pos = ifaceId.find('_');
return ifaceId.substr(0, pos);
}
inline void
parseInterfaceData(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
const std::string& ifaceId,
const EthernetInterfaceData& ethData,
const std::vector<IPv4AddressData>& ipv4Data,
const std::vector<IPv6AddressData>& ipv6Data,
const std::vector<StaticGatewayData>& ipv6GatewayData)
{
nlohmann::json& jsonResponse = asyncResp->res.jsonValue;
jsonResponse["Id"] = ifaceId;
jsonResponse["@odata.id"] = boost::urls::format(
"/redfish/v1/Managers/bmc/EthernetInterfaces/{}", ifaceId);
jsonResponse["InterfaceEnabled"] = ethData.nicEnabled;
if (ethData.nicEnabled)
{
jsonResponse["LinkStatus"] = ethData.linkUp ? "LinkUp" : "LinkDown";
jsonResponse["Status"]["State"] = "Enabled";
}
else
{
jsonResponse["LinkStatus"] = "NoLink";
jsonResponse["Status"]["State"] = "Disabled";
}
jsonResponse["SpeedMbps"] = ethData.speed;
jsonResponse["MTUSize"] = ethData.mtuSize;
jsonResponse["MACAddress"] = ethData.macAddress;
jsonResponse["DHCPv4"]["DHCPEnabled"] =
translateDhcpEnabledToBool(ethData.dhcpEnabled, true);
jsonResponse["DHCPv4"]["UseNTPServers"] = ethData.ntpv4Enabled;
jsonResponse["DHCPv4"]["UseDNSServers"] = ethData.dnsv4Enabled;
jsonResponse["DHCPv4"]["UseDomainName"] = ethData.hostNamev4Enabled;
jsonResponse["DHCPv6"]["OperatingMode"] =
translateDhcpEnabledToBool(ethData.dhcpEnabled, false) ? "Enabled"
: "Disabled";
jsonResponse["DHCPv6"]["UseNTPServers"] = ethData.ntpv6Enabled;
jsonResponse["DHCPv6"]["UseDNSServers"] = ethData.dnsv6Enabled;
jsonResponse["DHCPv6"]["UseDomainName"] = ethData.hostNamev6Enabled;
jsonResponse["StatelessAddressAutoConfig"]["IPv6AutoConfigEnabled"] =
ethData.ipv6AcceptRa;
if (!ethData.hostName.empty())
{
jsonResponse["HostName"] = ethData.hostName;
// When domain name is empty then it means, that it is a network
// without domain names, and the host name itself must be treated as
// FQDN
std::string fqdn = ethData.hostName;
if (!ethData.domainnames.empty())
{
fqdn += "." + ethData.domainnames[0];
}
jsonResponse["FQDN"] = fqdn;
}
if (ethData.vlanId)
{
jsonResponse["EthernetInterfaceType"] = "Virtual";
jsonResponse["VLAN"]["VLANEnable"] = true;
jsonResponse["VLAN"]["VLANId"] = *ethData.vlanId;
jsonResponse["VLAN"]["Tagged"] = true;
nlohmann::json::array_t relatedInterfaces;
nlohmann::json& parentInterface = relatedInterfaces.emplace_back();
parentInterface["@odata.id"] =
boost::urls::format("/redfish/v1/Managers/bmc/EthernetInterfaces",
extractParentInterfaceName(ifaceId));
jsonResponse["Links"]["RelatedInterfaces"] =
std::move(relatedInterfaces);
}
else
{
jsonResponse["EthernetInterfaceType"] = "Physical";
}
jsonResponse["NameServers"] = ethData.nameServers;
jsonResponse["StaticNameServers"] = ethData.staticNameServers;
nlohmann::json& ipv4Array = jsonResponse["IPv4Addresses"];
nlohmann::json& ipv4StaticArray = jsonResponse["IPv4StaticAddresses"];
ipv4Array = nlohmann::json::array();
ipv4StaticArray = nlohmann::json::array();
for (const auto& ipv4Config : ipv4Data)
{
std::string gatewayStr = ipv4Config.gateway;
if (gatewayStr.empty())
{
gatewayStr = "0.0.0.0";
}
nlohmann::json::object_t ipv4;
ipv4["AddressOrigin"] = ipv4Config.origin;
ipv4["SubnetMask"] = ipv4Config.netmask;
ipv4["Address"] = ipv4Config.address;
ipv4["Gateway"] = gatewayStr;
if (ipv4Config.origin == "Static")
{
ipv4StaticArray.push_back(ipv4);
}
ipv4Array.emplace_back(std::move(ipv4));
}
std::string ipv6GatewayStr = ethData.ipv6DefaultGateway;
if (ipv6GatewayStr.empty())
{
ipv6GatewayStr = "0:0:0:0:0:0:0:0";
}
jsonResponse["IPv6DefaultGateway"] = ipv6GatewayStr;
nlohmann::json::array_t ipv6StaticGatewayArray;
for (const auto& ipv6GatewayConfig : ipv6GatewayData)
{
nlohmann::json::object_t ipv6Gateway;
ipv6Gateway["Address"] = ipv6GatewayConfig.gateway;
ipv6Gateway["PrefixLength"] = ipv6GatewayConfig.prefixLength;
ipv6StaticGatewayArray.emplace_back(std::move(ipv6Gateway));
}
jsonResponse["IPv6StaticDefaultGateways"] =
std::move(ipv6StaticGatewayArray);
nlohmann::json& ipv6Array = jsonResponse["IPv6Addresses"];
nlohmann::json& ipv6StaticArray = jsonResponse["IPv6StaticAddresses"];
ipv6Array = nlohmann::json::array();
ipv6StaticArray = nlohmann::json::array();
nlohmann::json& ipv6AddrPolicyTable =
jsonResponse["IPv6AddressPolicyTable"];
ipv6AddrPolicyTable = nlohmann::json::array();
for (const auto& ipv6Config : ipv6Data)
{
nlohmann::json::object_t ipv6;
ipv6["Address"] = ipv6Config.address;
ipv6["PrefixLength"] = ipv6Config.prefixLength;
ipv6["AddressOrigin"] = ipv6Config.origin;
ipv6Array.emplace_back(std::move(ipv6));
if (ipv6Config.origin == "Static")
{
nlohmann::json::object_t ipv6Static;
ipv6Static["Address"] = ipv6Config.address;
ipv6Static["PrefixLength"] = ipv6Config.prefixLength;
ipv6StaticArray.emplace_back(std::move(ipv6Static));
}
}
}
inline void afterDelete(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
const std::string& ifaceId,
const boost::system::error_code& ec,
const sdbusplus::message_t& m)
{
if (!ec)
{
return;
}
const sd_bus_error* dbusError = m.get_error();
if (dbusError == nullptr)
{
messages::internalError(asyncResp->res);
return;
}
BMCWEB_LOG_DEBUG("DBus error: {}", dbusError->name);
if (std::string_view("org.freedesktop.DBus.Error.UnknownObject") ==
dbusError->name)
{
messages::resourceNotFound(asyncResp->res, "EthernetInterface",
ifaceId);
return;
}
if (std::string_view("org.freedesktop.DBus.Error.UnknownMethod") ==
dbusError->name)
{
messages::resourceCannotBeDeleted(asyncResp->res);
return;
}
messages::internalError(asyncResp->res);
}
inline void afterVlanCreate(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
const std::string& parentInterfaceUri,
const std::string& vlanInterface,
const boost::system::error_code& ec,
const sdbusplus::message_t& m
)
{
if (ec)
{
const sd_bus_error* dbusError = m.get_error();
if (dbusError == nullptr)
{
messages::internalError(asyncResp->res);
return;
}
BMCWEB_LOG_DEBUG("DBus error: {}", dbusError->name);
if (std::string_view(
"xyz.openbmc_project.Common.Error.ResourceNotFound") ==
dbusError->name)
{
messages::propertyValueNotInList(
asyncResp->res, parentInterfaceUri,
"Links/RelatedInterfaces/0/@odata.id");
return;
}
if (std::string_view(
"xyz.openbmc_project.Common.Error.InvalidArgument") ==
dbusError->name)
{
messages::resourceAlreadyExists(asyncResp->res, "EthernetInterface",
"Id", vlanInterface);
return;
}
messages::internalError(asyncResp->res);
return;
}
const boost::urls::url vlanInterfaceUri = boost::urls::format(
"/redfish/v1/Managers/bmc/EthernetInterfaces/{}", vlanInterface);
asyncResp->res.addHeader("Location", vlanInterfaceUri.buffer());
}
inline void requestEthernetInterfacesRoutes(App& app)
{
BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/EthernetInterfaces/")
.privileges(redfish::privileges::getEthernetInterfaceCollection)
.methods(boost::beast::http::verb::get)(
[&app](const crow::Request& req,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
if (!redfish::setUpRedfishRoute(app, req, asyncResp))
{
return;
}
asyncResp->res.jsonValue["@odata.type"] =
"#EthernetInterfaceCollection.EthernetInterfaceCollection";
asyncResp->res.jsonValue["@odata.id"] =
"/redfish/v1/Managers/bmc/EthernetInterfaces";
asyncResp->res.jsonValue["Name"] =
"Ethernet Network Interface Collection";
asyncResp->res.jsonValue["Description"] =
"Collection of EthernetInterfaces for this Manager";
// Get eth interface list, and call the below callback for JSON
// preparation
getEthernetIfaceList(
[asyncResp](const bool& success,
const std::vector<std::string>& ifaceList) {
if (!success)
{
messages::internalError(asyncResp->res);
return;
}
nlohmann::json& ifaceArray = asyncResp->res.jsonValue["Members"];
ifaceArray = nlohmann::json::array();
for (const std::string& ifaceItem : ifaceList)
{
nlohmann::json::object_t iface;
iface["@odata.id"] = boost::urls::format(
"/redfish/v1/Managers/bmc/EthernetInterfaces/{}",
ifaceItem);
ifaceArray.push_back(std::move(iface));
}
asyncResp->res.jsonValue["Members@odata.count"] = ifaceArray.size();
asyncResp->res.jsonValue["@odata.id"] =
"/redfish/v1/Managers/bmc/EthernetInterfaces";
});
});
BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/EthernetInterfaces/")
.privileges(redfish::privileges::postEthernetInterfaceCollection)
.methods(boost::beast::http::verb::post)(
[&app](const crow::Request& req,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) {
if (!redfish::setUpRedfishRoute(app, req, asyncResp))
{
return;
}
bool vlanEnable = false;
uint32_t vlanId = 0;
std::vector<nlohmann::json::object_t> relatedInterfaces;
if (!json_util::readJsonPatch(req, asyncResp->res, "VLAN/VLANEnable",
vlanEnable, "VLAN/VLANId", vlanId,
"Links/RelatedInterfaces",
relatedInterfaces))
{
return;
}
if (relatedInterfaces.size() != 1)
{
messages::arraySizeTooLong(asyncResp->res,
"Links/RelatedInterfaces",
relatedInterfaces.size());
return;
}
std::string parentInterfaceUri;
if (!json_util::readJsonObject(relatedInterfaces[0], asyncResp->res,
"@odata.id", parentInterfaceUri))
{
messages::propertyMissing(asyncResp->res,
"Links/RelatedInterfaces/0/@odata.id");
return;
}
BMCWEB_LOG_INFO("Parent Interface URI: {}", parentInterfaceUri);
boost::system::result<boost::urls::url_view> parsedUri =
boost::urls::parse_relative_ref(parentInterfaceUri);
if (!parsedUri)
{
messages::propertyValueFormatError(
asyncResp->res, parentInterfaceUri,
"Links/RelatedInterfaces/0/@odata.id");
return;
}
std::string parentInterface;
if (!crow::utility::readUrlSegments(
*parsedUri, "redfish", "v1", "Managers", "bmc",
"EthernetInterfaces", std::ref(parentInterface)))
{
messages::propertyValueNotInList(
asyncResp->res, parentInterfaceUri,
"Links/RelatedInterfaces/0/@odata.id");
return;
}
if (!vlanEnable)
{
// In OpenBMC implementation, VLANEnable cannot be false on
// create
messages::propertyValueIncorrect(asyncResp->res, "VLAN/VLANEnable",
"false");
return;
}
std::string vlanInterface = parentInterface + "_" +
std::to_string(vlanId);
crow::connections::systemBus->async_method_call(
[asyncResp, parentInterfaceUri,
vlanInterface](const boost::system::error_code& ec,
const sdbusplus::message_t& m) {
afterVlanCreate(asyncResp, parentInterfaceUri, vlanInterface, ec,
m);
},
"xyz.openbmc_project.Network", "/xyz/openbmc_project/network",
"xyz.openbmc_project.Network.VLAN.Create", "VLAN", parentInterface,
vlanId);
});
BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/EthernetInterfaces/<str>/")
.privileges(redfish::privileges::getEthernetInterface)
.methods(boost::beast::http::verb::get)(
[&app](const crow::Request& req,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
const std::string& ifaceId) {
if (!redfish::setUpRedfishRoute(app, req, asyncResp))
{
return;
}
getEthernetIfaceData(
ifaceId,
[asyncResp,
ifaceId](const bool& success, const EthernetInterfaceData& ethData,
const std::vector<IPv4AddressData>& ipv4Data,
const std::vector<IPv6AddressData>& ipv6Data,
const std::vector<StaticGatewayData>& ipv6GatewayData) {
if (!success)
{
// TODO(Pawel)consider distinguish between non
// existing object, and other errors
messages::resourceNotFound(asyncResp->res, "EthernetInterface",
ifaceId);
return;
}
asyncResp->res.jsonValue["@odata.type"] =
"#EthernetInterface.v1_9_0.EthernetInterface";
asyncResp->res.jsonValue["Name"] = "Manager Ethernet Interface";
asyncResp->res.jsonValue["Description"] =
"Management Network Interface";
parseInterfaceData(asyncResp, ifaceId, ethData, ipv4Data, ipv6Data,
ipv6GatewayData);
});
});
BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/EthernetInterfaces/<str>/")
.privileges(redfish::privileges::patchEthernetInterface)
.methods(boost::beast::http::verb::patch)(
[&app](const crow::Request& req,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
const std::string& ifaceId) {
if (!redfish::setUpRedfishRoute(app, req, asyncResp))
{
return;
}
std::optional<std::string> hostname;
std::optional<std::string> fqdn;
std::optional<std::string> macAddress;
std::optional<std::string> ipv6DefaultGateway;
std::optional<
std::vector<std::variant<nlohmann::json::object_t, std::nullptr_t>>>
ipv4StaticAddresses;
std::optional<
std::vector<std::variant<nlohmann::json::object_t, std::nullptr_t>>>
ipv6StaticAddresses;
std::optional<
std::vector<std::variant<nlohmann::json::object_t, std::nullptr_t>>>
ipv6StaticDefaultGateways;
std::optional<std::vector<std::string>> staticNameServers;
std::optional<bool> ipv6AutoConfigEnabled;
std::optional<bool> interfaceEnabled;
std::optional<size_t> mtuSize;
DHCPParameters v4dhcpParms;
DHCPParameters v6dhcpParms;
// clang-format off
if (!json_util::readJsonPatch(req, asyncResp->res,
"DHCPv4/DHCPEnabled", v4dhcpParms.dhcpv4Enabled,
"DHCPv4/UseDNSServers", v4dhcpParms.useDnsServers,
"DHCPv4/UseDomainName", v4dhcpParms.useDomainName,
"DHCPv4/UseNTPServers", v4dhcpParms.useNtpServers,
"DHCPv6/OperatingMode", v6dhcpParms.dhcpv6OperatingMode,
"DHCPv6/UseDNSServers", v6dhcpParms.useDnsServers,
"DHCPv6/UseDomainName", v6dhcpParms.useDomainName,
"DHCPv6/UseNTPServers", v6dhcpParms.useNtpServers,
"FQDN", fqdn,
"HostName", hostname,
"IPv4StaticAddresses", ipv4StaticAddresses,
"IPv6DefaultGateway", ipv6DefaultGateway,
"IPv6StaticAddresses", ipv6StaticAddresses,
"IPv6StaticDefaultGateways", ipv6StaticDefaultGateways,
"InterfaceEnabled", interfaceEnabled,
"MACAddress", macAddress,
"MTUSize", mtuSize,
"StatelessAddressAutoConfig/IPv6AutoConfigEnabled", ipv6AutoConfigEnabled,
"StaticNameServers", staticNameServers
)
)
{
return;
}
// clang-format on
// Get single eth interface data, and call the below callback
// for JSON preparation
getEthernetIfaceData(
ifaceId,
[asyncResp, ifaceId, hostname = std::move(hostname),
fqdn = std::move(fqdn), macAddress = std::move(macAddress),
ipv4StaticAddresses = std::move(ipv4StaticAddresses),
ipv6DefaultGateway = std::move(ipv6DefaultGateway),
ipv6StaticAddresses = std::move(ipv6StaticAddresses),
ipv6StaticDefaultGateway = std::move(ipv6StaticDefaultGateways),
staticNameServers = std::move(staticNameServers), mtuSize,
ipv6AutoConfigEnabled, v4dhcpParms = std::move(v4dhcpParms),
v6dhcpParms = std::move(v6dhcpParms), interfaceEnabled](
const bool success, const EthernetInterfaceData& ethData,
const std::vector<IPv4AddressData>& ipv4Data,
const std::vector<IPv6AddressData>& ipv6Data,
const std::vector<StaticGatewayData>& ipv6GatewayData) mutable {
if (!success)
{
// ... otherwise return error
// TODO(Pawel)consider distinguish between non
// existing object, and other errors
messages::resourceNotFound(asyncResp->res, "EthernetInterface",
ifaceId);
return;
}
handleDHCPPatch(ifaceId, ethData, v4dhcpParms, v6dhcpParms,
asyncResp);
if (hostname)
{
handleHostnamePatch(*hostname, asyncResp);
}
if (ipv6AutoConfigEnabled)
{
handleSLAACAutoConfigPatch(ifaceId, *ipv6AutoConfigEnabled,
asyncResp);
}
if (fqdn)
{
handleFqdnPatch(ifaceId, *fqdn, asyncResp);
}
if (macAddress)
{
handleMACAddressPatch(ifaceId, *macAddress, asyncResp);
}
if (ipv4StaticAddresses)
{
handleIPv4StaticPatch(ifaceId, *ipv4StaticAddresses, ethData,
ipv4Data, asyncResp);
}
if (staticNameServers)
{
handleStaticNameServersPatch(ifaceId, *staticNameServers,
asyncResp);
}
if (ipv6DefaultGateway)
{
messages::propertyNotWritable(asyncResp->res,
"IPv6DefaultGateway");
}
if (ipv6StaticAddresses)
{
handleIPv6StaticAddressesPatch(ifaceId, *ipv6StaticAddresses,
ipv6Data, asyncResp);
}
if (ipv6StaticDefaultGateway)
{
handleIPv6DefaultGateway(ifaceId, *ipv6StaticDefaultGateway,
ipv6GatewayData, asyncResp);
}
if (interfaceEnabled)
{
setDbusProperty(asyncResp, "xyz.openbmc_project.Network",
sdbusplus::message::object_path(
"/xyz/openbmc_project/network") /
ifaceId,
"xyz.openbmc_project.Network.EthernetInterface",
"NICEnabled", "InterfaceEnabled",
*interfaceEnabled);
}
if (mtuSize)
{
handleMTUSizePatch(ifaceId, *mtuSize, asyncResp);
}
});
});
BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/EthernetInterfaces/<str>/")
.privileges(redfish::privileges::deleteEthernetInterface)
.methods(boost::beast::http::verb::delete_)(
[&app](const crow::Request& req,
const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
const std::string& ifaceId) {
if (!redfish::setUpRedfishRoute(app, req, asyncResp))
{
return;
}
crow::connections::systemBus->async_method_call(
[asyncResp, ifaceId](const boost::system::error_code& ec,
const sdbusplus::message_t& m) {
afterDelete(asyncResp, ifaceId, ec, m);
},
"xyz.openbmc_project.Network",
std::string("/xyz/openbmc_project/network/") + ifaceId,
"xyz.openbmc_project.Object.Delete", "Delete");
});
}
} // namespace redfish