blob: ace2b1dc4bb6fe596d192e8ed9234e8a7ffb9270 [file] [log] [blame]
#include "config_parser.hpp"
#include "ipaddress.hpp"
#include "mock_network_manager.hpp"
#include "mock_syscall.hpp"
#include "util.hpp"
#include <arpa/inet.h>
#include <net/if.h>
#include <netinet/in.h>
#include <stdlib.h>
#include <charconv>
#include <exception>
#include <sdbusplus/bus.hpp>
#include <stdplus/gtest/tmp.hpp>
#include <string_view>
#include <xyz/openbmc_project/Common/error.hpp>
#include <gtest/gtest.h>
namespace phosphor
{
namespace network
{
using std::literals::string_view_literals::operator""sv;
class TestEthernetInterface : public stdplus::gtest::TestWithTmp
{
public:
sdbusplus::bus_t bus;
std::string confDir;
MockManager manager;
MockEthernetInterface interface;
TestEthernetInterface() :
bus(sdbusplus::bus::new_default()), confDir(CaseTmpDir()),
manager(bus, "/xyz/openbmc_test/network", confDir),
interface(makeInterface(bus, manager))
{
}
static MockEthernetInterface makeInterface(sdbusplus::bus_t& bus,
MockManager& manager)
{
mock_clear();
mock_addIF("test0", /*idx=*/1);
return {bus, manager, getInterfaceInfo("test0"),
"/xyz/openbmc_test/network"sv, config::Parser()};
}
int countIPObjects()
{
return interface.getAddresses().size();
}
bool isIPObjectExist(const std::string& ipaddress)
{
auto address = interface.getAddresses().find(ipaddress);
if (address == interface.getAddresses().end())
{
return false;
}
return true;
}
bool deleteIPObject(const std::string& ipaddress)
{
auto address = interface.getAddresses().find(ipaddress);
if (address == interface.getAddresses().end())
{
return false;
}
address->second->delete_();
return true;
}
std::string getObjectPath(const std::string& ipaddress, uint8_t subnetMask,
IP::AddressOrigin origin)
{
IP::Protocol addressType = IP::Protocol::IPv4;
return interface.generateObjectPath(addressType, ipaddress, subnetMask,
origin);
}
void createIPObject(IP::Protocol addressType, const std::string& ipaddress,
uint8_t subnetMask)
{
interface.ip(addressType, ipaddress, subnetMask, "");
}
void setNtpServers()
{
ServerList ntpServers = {"10.1.1.1", "10.2.2.2", "10.3.3.3"};
interface.EthernetInterfaceIntf::ntpServers(ntpServers);
}
ServerList getNtpServers()
{
return interface.EthernetInterfaceIntf::ntpServers();
}
};
TEST_F(TestEthernetInterface, Fields)
{
EXPECT_EQ(0, interface.mtu());
EXPECT_EQ("", interface.macAddress());
EXPECT_FALSE(interface.linkUp());
constexpr unsigned idx = 2;
constexpr ether_addr mac{0x11, 0x22, 0x33, 0x44, 0x55, 0x66};
constexpr unsigned mtu = 150;
mock_addIF("test1", idx, IFF_RUNNING, mac, mtu);
MockEthernetInterface intf(bus, manager, getInterfaceInfo("test1"),
"/xyz/openbmc_test/network"sv, config::Parser());
EXPECT_EQ(mtu, intf.mtu());
EXPECT_EQ(std::to_string(mac), intf.macAddress());
EXPECT_TRUE(intf.linkUp());
}
TEST_F(TestEthernetInterface, NoIPaddress)
{
EXPECT_EQ(countIPObjects(), 0);
}
TEST_F(TestEthernetInterface, AddIPAddress)
{
IP::Protocol addressType = IP::Protocol::IPv4;
createIPObject(addressType, "10.10.10.10", 16);
EXPECT_EQ(true, isIPObjectExist("10.10.10.10"));
}
TEST_F(TestEthernetInterface, AddMultipleAddress)
{
IP::Protocol addressType = IP::Protocol::IPv4;
createIPObject(addressType, "10.10.10.10", 16);
createIPObject(addressType, "20.20.20.20", 16);
EXPECT_EQ(true, isIPObjectExist("10.10.10.10"));
EXPECT_EQ(true, isIPObjectExist("20.20.20.20"));
}
TEST_F(TestEthernetInterface, DeleteIPAddress)
{
IP::Protocol addressType = IP::Protocol::IPv4;
createIPObject(addressType, "10.10.10.10", 16);
createIPObject(addressType, "20.20.20.20", 16);
deleteIPObject("10.10.10.10");
EXPECT_EQ(false, isIPObjectExist("10.10.10.10"));
EXPECT_EQ(true, isIPObjectExist("20.20.20.20"));
}
TEST_F(TestEthernetInterface, DeleteInvalidIPAddress)
{
EXPECT_EQ(false, deleteIPObject("10.10.10.10"));
}
TEST_F(TestEthernetInterface, CheckObjectPath)
{
std::string ipaddress = "10.10.10.10";
uint8_t prefix = 16;
IP::AddressOrigin origin = IP::AddressOrigin::Static;
auto path = getObjectPath(ipaddress, prefix, origin);
auto pathsv = std::string_view(path);
constexpr auto expectedPrefix = "/xyz/openbmc_test/network/test0/ipv4/"sv;
EXPECT_TRUE(pathsv.starts_with(expectedPrefix));
pathsv.remove_prefix(expectedPrefix.size());
uint32_t val;
auto [ptr, res] = std::from_chars(pathsv.begin(), pathsv.end(), val, 16);
EXPECT_EQ(res, std::errc());
EXPECT_EQ(ptr, pathsv.end());
EXPECT_EQ(path, getObjectPath(ipaddress, prefix, origin));
origin = IP::AddressOrigin::DHCP;
EXPECT_NE(path, getObjectPath(ipaddress, prefix, origin));
}
TEST_F(TestEthernetInterface, addStaticNameServers)
{
ServerList servers = {"9.1.1.1", "9.2.2.2", "9.3.3.3"};
EXPECT_CALL(manager, reloadConfigs());
interface.staticNameServers(servers);
fs::path filePath = confDir;
filePath /= "00-bmc-test0.network";
config::Parser parser(filePath.string());
EXPECT_EQ(servers, parser.map.getValueStrings("Network", "DNS"));
}
TEST_F(TestEthernetInterface, getDynamicNameServers)
{
ServerList servers = {"9.1.1.1", "9.2.2.2", "9.3.3.3"};
EXPECT_CALL(interface, getNameServerFromResolvd())
.WillRepeatedly(testing::Return(servers));
EXPECT_EQ(interface.getNameServerFromResolvd(), servers);
}
TEST_F(TestEthernetInterface, addStaticNTPServers)
{
ServerList servers = {"10.1.1.1", "10.2.2.2", "10.3.3.3"};
EXPECT_CALL(manager, reloadConfigs());
interface.staticNTPServers(servers);
fs::path filePath = confDir;
filePath /= "00-bmc-test0.network";
config::Parser parser(filePath.string());
EXPECT_EQ(servers, parser.map.getValueStrings("Network", "NTP"));
}
TEST_F(TestEthernetInterface, addNTPServers)
{
using namespace sdbusplus::xyz::openbmc_project::Common::Error;
ServerList servers = {"10.1.1.1", "10.2.2.2", "10.3.3.3"};
EXPECT_THROW(interface.ntpServers(servers), NotAllowed);
}
TEST_F(TestEthernetInterface, getNTPServers)
{
ServerList servers = {"10.1.1.1", "10.2.2.2", "10.3.3.3"};
setNtpServers();
EXPECT_EQ(getNtpServers(), servers);
}
TEST_F(TestEthernetInterface, addGateway)
{
std::string gateway = "10.3.3.3";
interface.defaultGateway(gateway);
EXPECT_EQ(interface.defaultGateway(), gateway);
interface.defaultGateway("");
EXPECT_EQ(interface.defaultGateway(), "");
}
TEST_F(TestEthernetInterface, addGateway6)
{
std::string gateway6 = "ffff:ffff:ffff:fe80::1";
interface.defaultGateway6(gateway6);
EXPECT_EQ(interface.defaultGateway6(), gateway6);
interface.defaultGateway6("");
EXPECT_EQ(interface.defaultGateway6(), "");
}
TEST_F(TestEthernetInterface, DHCPEnabled)
{
EXPECT_CALL(manager, reloadConfigs()).WillRepeatedly(testing::Return());
using DHCPConf = EthernetInterfaceIntf::DHCPConf;
auto test = [&](DHCPConf conf, bool dhcp4, bool dhcp6, bool ra) {
EXPECT_EQ(conf, interface.dhcpEnabled());
EXPECT_EQ(dhcp4, interface.dhcp4());
EXPECT_EQ(dhcp6, interface.dhcp6());
EXPECT_EQ(ra, interface.ipv6AcceptRA());
};
test(DHCPConf::both, /*dhcp4=*/true, /*dhcp6=*/true, /*ra=*/true);
auto set_test = [&](DHCPConf conf, bool dhcp4, bool dhcp6, bool ra) {
EXPECT_EQ(conf, interface.dhcpEnabled(conf));
test(conf, dhcp4, dhcp6, ra);
};
set_test(DHCPConf::none, /*dhcp4=*/false, /*dhcp6=*/false, /*ra=*/false);
set_test(DHCPConf::v4, /*dhcp4=*/true, /*dhcp6=*/false, /*ra=*/false);
set_test(DHCPConf::v6stateless, /*dhcp4=*/false, /*dhcp6=*/false,
/*ra=*/true);
set_test(DHCPConf::v6, /*dhcp4=*/false, /*dhcp6=*/true, /*ra=*/true);
set_test(DHCPConf::v4v6stateless, /*dhcp4=*/true, /*dhcp6=*/false,
/*ra=*/true);
set_test(DHCPConf::both, /*dhcp4=*/true, /*dhcp6=*/true, /*ra=*/true);
auto ind_test = [&](DHCPConf conf, bool dhcp4, bool dhcp6, bool ra) {
EXPECT_EQ(dhcp4, interface.dhcp4(dhcp4));
EXPECT_EQ(dhcp6, interface.dhcp6(dhcp6));
EXPECT_EQ(ra, interface.ipv6AcceptRA(ra));
test(conf, dhcp4, dhcp6, ra);
};
ind_test(DHCPConf::none, /*dhcp4=*/false, /*dhcp6=*/false, /*ra=*/false);
ind_test(DHCPConf::v4, /*dhcp4=*/true, /*dhcp6=*/false, /*ra=*/false);
ind_test(DHCPConf::v6stateless, /*dhcp4=*/false, /*dhcp6=*/false,
/*ra=*/true);
ind_test(DHCPConf::v6, /*dhcp4=*/false, /*dhcp6=*/true, /*ra=*/false);
set_test(DHCPConf::v6, /*dhcp4=*/false, /*dhcp6=*/true, /*ra=*/true);
ind_test(DHCPConf::v4v6stateless, /*dhcp4=*/true, /*dhcp6=*/false,
/*ra=*/true);
ind_test(DHCPConf::both, /*dhcp4=*/true, /*dhcp6=*/true, /*ra=*/false);
set_test(DHCPConf::both, /*dhcp4=*/true, /*dhcp6=*/true, /*ra=*/true);
}
} // namespace network
} // namespace phosphor