428 lines
15 KiB
C++
428 lines
15 KiB
C++
|
|
// Copyright 2012 The Chromium Authors
|
||
|
|
// Use of this source code is governed by a BSD-style license that can be
|
||
|
|
// found in the LICENSE file.
|
||
|
|
|
||
|
|
#include "net/base/ip_endpoint.h"
|
||
|
|
|
||
|
|
#include <string.h>
|
||
|
|
|
||
|
|
#include <string>
|
||
|
|
#include <tuple>
|
||
|
|
|
||
|
|
#include "base/check_op.h"
|
||
|
|
#include "base/notreached.h"
|
||
|
|
#include "base/numerics/safe_conversions.h"
|
||
|
|
#include "base/strings/string_number_conversions.h"
|
||
|
|
#include "base/sys_byteorder.h"
|
||
|
|
#include "base/values.h"
|
||
|
|
#include "build/build_config.h"
|
||
|
|
#include "net/base/ip_address.h"
|
||
|
|
#include "net/base/sockaddr_storage.h"
|
||
|
|
#include "net/base/sys_addrinfo.h"
|
||
|
|
#include "testing/gmock/include/gmock/gmock.h"
|
||
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
||
|
|
#include "testing/platform_test.h"
|
||
|
|
#include "third_party/abseil-cpp/absl/types/optional.h"
|
||
|
|
|
||
|
|
#if BUILDFLAG(IS_WIN)
|
||
|
|
#include <winsock2.h>
|
||
|
|
#include <ws2bth.h>
|
||
|
|
|
||
|
|
#include "base/test/gtest_util.h" // For EXPECT_DCHECK_DEATH
|
||
|
|
#include "net/base/winsock_util.h" // For kBluetoothAddressSize
|
||
|
|
#elif BUILDFLAG(IS_POSIX)
|
||
|
|
#include <netinet/in.h>
|
||
|
|
#endif
|
||
|
|
|
||
|
|
using testing::Optional;
|
||
|
|
|
||
|
|
namespace net {
|
||
|
|
|
||
|
|
namespace {
|
||
|
|
|
||
|
|
// Retuns the port field of the |sockaddr|.
|
||
|
|
const uint16_t* GetPortFieldFromSockaddr(const struct sockaddr* address,
|
||
|
|
socklen_t address_len) {
|
||
|
|
if (address->sa_family == AF_INET) {
|
||
|
|
DCHECK_LE(sizeof(sockaddr_in), static_cast<size_t>(address_len));
|
||
|
|
const struct sockaddr_in* sockaddr =
|
||
|
|
reinterpret_cast<const struct sockaddr_in*>(address);
|
||
|
|
return &sockaddr->sin_port;
|
||
|
|
} else if (address->sa_family == AF_INET6) {
|
||
|
|
DCHECK_LE(sizeof(sockaddr_in6), static_cast<size_t>(address_len));
|
||
|
|
const struct sockaddr_in6* sockaddr =
|
||
|
|
reinterpret_cast<const struct sockaddr_in6*>(address);
|
||
|
|
return &sockaddr->sin6_port;
|
||
|
|
} else {
|
||
|
|
NOTREACHED();
|
||
|
|
return nullptr;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
// Returns the value of port in |sockaddr| (in host byte ordering).
|
||
|
|
int GetPortFromSockaddr(const struct sockaddr* address, socklen_t address_len) {
|
||
|
|
const uint16_t* port_field = GetPortFieldFromSockaddr(address, address_len);
|
||
|
|
if (!port_field)
|
||
|
|
return -1;
|
||
|
|
return base::NetToHost16(*port_field);
|
||
|
|
}
|
||
|
|
|
||
|
|
struct TestData {
|
||
|
|
std::string host;
|
||
|
|
std::string host_normalized;
|
||
|
|
bool ipv6;
|
||
|
|
IPAddress ip_address;
|
||
|
|
} tests[] = {
|
||
|
|
{"127.0.00.1", "127.0.0.1", false},
|
||
|
|
{"192.168.1.1", "192.168.1.1", false},
|
||
|
|
{"::1", "[::1]", true},
|
||
|
|
{"2001:db8:0::42", "[2001:db8::42]", true},
|
||
|
|
};
|
||
|
|
|
||
|
|
class IPEndPointTest : public PlatformTest {
|
||
|
|
public:
|
||
|
|
void SetUp() override {
|
||
|
|
// This is where we populate the TestData.
|
||
|
|
for (auto& test : tests) {
|
||
|
|
EXPECT_TRUE(test.ip_address.AssignFromIPLiteral(test.host));
|
||
|
|
}
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, Constructor) {
|
||
|
|
{
|
||
|
|
IPEndPoint endpoint;
|
||
|
|
EXPECT_EQ(0, endpoint.port());
|
||
|
|
}
|
||
|
|
|
||
|
|
for (const auto& test : tests) {
|
||
|
|
IPEndPoint endpoint(test.ip_address, 80);
|
||
|
|
EXPECT_EQ(80, endpoint.port());
|
||
|
|
EXPECT_EQ(test.ip_address, endpoint.address());
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, Assignment) {
|
||
|
|
uint16_t port = 0;
|
||
|
|
for (const auto& test : tests) {
|
||
|
|
IPEndPoint src(test.ip_address, ++port);
|
||
|
|
IPEndPoint dest = src;
|
||
|
|
|
||
|
|
EXPECT_EQ(src.port(), dest.port());
|
||
|
|
EXPECT_EQ(src.address(), dest.address());
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, Copy) {
|
||
|
|
uint16_t port = 0;
|
||
|
|
for (const auto& test : tests) {
|
||
|
|
IPEndPoint src(test.ip_address, ++port);
|
||
|
|
IPEndPoint dest(src);
|
||
|
|
|
||
|
|
EXPECT_EQ(src.port(), dest.port());
|
||
|
|
EXPECT_EQ(src.address(), dest.address());
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, ToFromSockAddr) {
|
||
|
|
uint16_t port = 0;
|
||
|
|
for (const auto& test : tests) {
|
||
|
|
IPEndPoint ip_endpoint(test.ip_address, ++port);
|
||
|
|
|
||
|
|
// Convert to a sockaddr.
|
||
|
|
SockaddrStorage storage;
|
||
|
|
EXPECT_TRUE(ip_endpoint.ToSockAddr(storage.addr, &storage.addr_len));
|
||
|
|
|
||
|
|
// Basic verification.
|
||
|
|
socklen_t expected_size =
|
||
|
|
test.ipv6 ? sizeof(struct sockaddr_in6) : sizeof(struct sockaddr_in);
|
||
|
|
EXPECT_EQ(expected_size, storage.addr_len);
|
||
|
|
EXPECT_EQ(ip_endpoint.port(),
|
||
|
|
GetPortFromSockaddr(storage.addr, storage.addr_len));
|
||
|
|
// And convert back to an IPEndPoint.
|
||
|
|
IPEndPoint ip_endpoint2;
|
||
|
|
EXPECT_TRUE(ip_endpoint2.FromSockAddr(storage.addr, storage.addr_len));
|
||
|
|
EXPECT_EQ(ip_endpoint.port(), ip_endpoint2.port());
|
||
|
|
EXPECT_EQ(ip_endpoint.address(), ip_endpoint2.address());
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, ToSockAddrBufTooSmall) {
|
||
|
|
uint16_t port = 0;
|
||
|
|
for (const auto& test : tests) {
|
||
|
|
IPEndPoint ip_endpoint(test.ip_address, port);
|
||
|
|
|
||
|
|
SockaddrStorage storage;
|
||
|
|
storage.addr_len = 3; // size is too small!
|
||
|
|
EXPECT_FALSE(ip_endpoint.ToSockAddr(storage.addr, &storage.addr_len));
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, FromSockAddrBufTooSmall) {
|
||
|
|
struct sockaddr_in addr;
|
||
|
|
memset(&addr, 0, sizeof(addr));
|
||
|
|
addr.sin_family = AF_INET;
|
||
|
|
IPEndPoint ip_endpoint;
|
||
|
|
struct sockaddr* sockaddr = reinterpret_cast<struct sockaddr*>(&addr);
|
||
|
|
EXPECT_FALSE(ip_endpoint.FromSockAddr(sockaddr, sizeof(addr) - 1));
|
||
|
|
}
|
||
|
|
|
||
|
|
#if BUILDFLAG(IS_WIN)
|
||
|
|
|
||
|
|
namespace {
|
||
|
|
constexpr uint8_t kBluetoothAddrBytes[kBluetoothAddressSize] = {1, 2, 3,
|
||
|
|
4, 5, 6};
|
||
|
|
constexpr uint8_t kBluetoothAddrBytes2[kBluetoothAddressSize] = {1, 2, 3,
|
||
|
|
4, 5, 7};
|
||
|
|
const IPAddress kBluetoothAddress(kBluetoothAddrBytes);
|
||
|
|
const IPAddress kBluetoothAddress2(kBluetoothAddrBytes2);
|
||
|
|
|
||
|
|
// Select a Bluetooth port that does not fit in a uint16_t.
|
||
|
|
constexpr uint32_t kBluetoothPort = std::numeric_limits<uint16_t>::max() + 1;
|
||
|
|
|
||
|
|
SOCKADDR_BTH BuildBluetoothSockAddr(const IPAddress& ip_address,
|
||
|
|
uint32_t port) {
|
||
|
|
SOCKADDR_BTH addr = {};
|
||
|
|
addr.addressFamily = AF_BTH;
|
||
|
|
DCHECK_LE(ip_address.bytes().size(), sizeof(addr.btAddr));
|
||
|
|
memcpy(&addr.btAddr, ip_address.bytes().data(), ip_address.bytes().size());
|
||
|
|
addr.port = port;
|
||
|
|
return addr;
|
||
|
|
}
|
||
|
|
} // namespace
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, WinBluetoothSockAddrCompareWithSelf) {
|
||
|
|
IPEndPoint bt_endpoint;
|
||
|
|
SOCKADDR_BTH addr = BuildBluetoothSockAddr(kBluetoothAddress, kBluetoothPort);
|
||
|
|
EXPECT_TRUE(bt_endpoint.FromSockAddr(
|
||
|
|
reinterpret_cast<const struct sockaddr*>(&addr), sizeof(addr)));
|
||
|
|
EXPECT_EQ(bt_endpoint.address(), kBluetoothAddress);
|
||
|
|
EXPECT_EQ(bt_endpoint.GetFamily(), AddressFamily::ADDRESS_FAMILY_UNSPECIFIED);
|
||
|
|
EXPECT_EQ(bt_endpoint.GetSockAddrFamily(), AF_BTH);
|
||
|
|
// Comparison functions should agree that `bt_endpoint` equals itself.
|
||
|
|
EXPECT_FALSE(bt_endpoint < bt_endpoint);
|
||
|
|
EXPECT_FALSE(bt_endpoint != bt_endpoint);
|
||
|
|
EXPECT_TRUE(bt_endpoint == bt_endpoint);
|
||
|
|
// Test that IPv4/IPv6-only methods crash.
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint.port());
|
||
|
|
SockaddrStorage storage;
|
||
|
|
EXPECT_DCHECK_DEATH(
|
||
|
|
std::ignore = bt_endpoint.ToSockAddr(storage.addr, &storage.addr_len));
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint.ToString());
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint.ToStringWithoutPort());
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, WinBluetoothSockAddrCompareWithNonBluetooth) {
|
||
|
|
IPEndPoint bt_endpoint;
|
||
|
|
SOCKADDR_BTH addr = BuildBluetoothSockAddr(kBluetoothAddress, kBluetoothPort);
|
||
|
|
EXPECT_TRUE(bt_endpoint.FromSockAddr(
|
||
|
|
reinterpret_cast<const struct sockaddr*>(&addr), sizeof(addr)));
|
||
|
|
|
||
|
|
// Compare `bt_endpoint` with non-Bluetooth endpoints.
|
||
|
|
for (const auto& test : tests) {
|
||
|
|
IPEndPoint endpoint(test.ip_address, 80);
|
||
|
|
if (test.ip_address.IsIPv4()) {
|
||
|
|
EXPECT_FALSE(bt_endpoint < endpoint);
|
||
|
|
} else {
|
||
|
|
EXPECT_TRUE(test.ip_address.IsIPv6());
|
||
|
|
EXPECT_TRUE(bt_endpoint < endpoint);
|
||
|
|
}
|
||
|
|
EXPECT_TRUE(bt_endpoint != endpoint);
|
||
|
|
EXPECT_FALSE(bt_endpoint == endpoint);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, WinBluetoothSockAddrCompareWithCopy) {
|
||
|
|
IPEndPoint bt_endpoint;
|
||
|
|
SOCKADDR_BTH addr = BuildBluetoothSockAddr(kBluetoothAddress, kBluetoothPort);
|
||
|
|
EXPECT_TRUE(bt_endpoint.FromSockAddr(
|
||
|
|
reinterpret_cast<const struct sockaddr*>(&addr), sizeof(addr)));
|
||
|
|
|
||
|
|
// Verify that a copy's accessors return the same values as the original's.
|
||
|
|
IPEndPoint bt_endpoint_other(bt_endpoint);
|
||
|
|
EXPECT_EQ(bt_endpoint.address(), bt_endpoint_other.address());
|
||
|
|
EXPECT_EQ(bt_endpoint.GetFamily(), bt_endpoint_other.GetFamily());
|
||
|
|
EXPECT_EQ(bt_endpoint.GetSockAddrFamily(),
|
||
|
|
bt_endpoint_other.GetSockAddrFamily());
|
||
|
|
// Comparison functions should agree that the endpoints are equal.
|
||
|
|
EXPECT_FALSE(bt_endpoint < bt_endpoint_other);
|
||
|
|
EXPECT_FALSE(bt_endpoint != bt_endpoint_other);
|
||
|
|
EXPECT_TRUE(bt_endpoint == bt_endpoint_other);
|
||
|
|
// Test that IPv4/IPv6-only methods crash.
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint_other.port());
|
||
|
|
SockaddrStorage storage;
|
||
|
|
EXPECT_DCHECK_DEATH(std::ignore = bt_endpoint_other.ToSockAddr(
|
||
|
|
storage.addr, &storage.addr_len));
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint_other.ToString());
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint_other.ToStringWithoutPort());
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, WinBluetoothSockAddrCompareWithDifferentPort) {
|
||
|
|
IPEndPoint bt_endpoint;
|
||
|
|
SOCKADDR_BTH addr = BuildBluetoothSockAddr(kBluetoothAddress, kBluetoothPort);
|
||
|
|
EXPECT_TRUE(bt_endpoint.FromSockAddr(
|
||
|
|
reinterpret_cast<const struct sockaddr*>(&addr), sizeof(addr)));
|
||
|
|
|
||
|
|
// Compare with another IPEndPoint that has a different port.
|
||
|
|
IPEndPoint bt_endpoint_other;
|
||
|
|
SOCKADDR_BTH addr2 =
|
||
|
|
BuildBluetoothSockAddr(kBluetoothAddress, kBluetoothPort + 1);
|
||
|
|
EXPECT_TRUE(bt_endpoint_other.FromSockAddr(
|
||
|
|
reinterpret_cast<const struct sockaddr*>(&addr2), sizeof(addr2)));
|
||
|
|
EXPECT_EQ(bt_endpoint.address(), bt_endpoint_other.address());
|
||
|
|
EXPECT_EQ(bt_endpoint.GetFamily(), bt_endpoint_other.GetFamily());
|
||
|
|
EXPECT_EQ(bt_endpoint.GetSockAddrFamily(),
|
||
|
|
bt_endpoint_other.GetSockAddrFamily());
|
||
|
|
// Comparison functions should agree that `bt_endpoint == bt_endpoint_other`
|
||
|
|
// because they have the same address and Bluetooth ports are not considered
|
||
|
|
// by comparison functions.
|
||
|
|
EXPECT_FALSE(bt_endpoint < bt_endpoint_other);
|
||
|
|
EXPECT_FALSE(bt_endpoint != bt_endpoint_other);
|
||
|
|
EXPECT_TRUE(bt_endpoint == bt_endpoint_other);
|
||
|
|
// Test that IPv4/IPv6-only methods crash.
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint_other.port());
|
||
|
|
SockaddrStorage storage;
|
||
|
|
EXPECT_DCHECK_DEATH(std::ignore = bt_endpoint_other.ToSockAddr(
|
||
|
|
storage.addr, &storage.addr_len));
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint_other.ToString());
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint_other.ToStringWithoutPort());
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, WinBluetoothSockAddrCompareWithDifferentAddress) {
|
||
|
|
IPEndPoint bt_endpoint;
|
||
|
|
SOCKADDR_BTH addr = BuildBluetoothSockAddr(kBluetoothAddress, kBluetoothPort);
|
||
|
|
EXPECT_TRUE(bt_endpoint.FromSockAddr(
|
||
|
|
reinterpret_cast<const struct sockaddr*>(&addr), sizeof(addr)));
|
||
|
|
|
||
|
|
// Compare with another IPEndPoint that has a different address.
|
||
|
|
IPEndPoint bt_endpoint_other;
|
||
|
|
SOCKADDR_BTH addr2 =
|
||
|
|
BuildBluetoothSockAddr(kBluetoothAddress2, kBluetoothPort);
|
||
|
|
EXPECT_TRUE(bt_endpoint_other.FromSockAddr(
|
||
|
|
reinterpret_cast<const struct sockaddr*>(&addr2), sizeof(addr2)));
|
||
|
|
EXPECT_LT(bt_endpoint.address(), bt_endpoint_other.address());
|
||
|
|
EXPECT_EQ(bt_endpoint.GetFamily(), bt_endpoint_other.GetFamily());
|
||
|
|
EXPECT_EQ(bt_endpoint.GetSockAddrFamily(),
|
||
|
|
bt_endpoint_other.GetSockAddrFamily());
|
||
|
|
// Comparison functions should agree that `bt_endpoint < bt_endpoint_other`
|
||
|
|
// due to lexicographic comparison of the address bytes.
|
||
|
|
EXPECT_TRUE(bt_endpoint < bt_endpoint_other);
|
||
|
|
EXPECT_TRUE(bt_endpoint != bt_endpoint_other);
|
||
|
|
EXPECT_FALSE(bt_endpoint == bt_endpoint_other);
|
||
|
|
// Test that IPv4/IPv6-only methods crash.
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint_other.port());
|
||
|
|
SockaddrStorage storage;
|
||
|
|
EXPECT_DCHECK_DEATH(std::ignore = bt_endpoint_other.ToSockAddr(
|
||
|
|
storage.addr, &storage.addr_len));
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint_other.ToString());
|
||
|
|
EXPECT_DCHECK_DEATH(bt_endpoint_other.ToStringWithoutPort());
|
||
|
|
}
|
||
|
|
#endif
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, Equality) {
|
||
|
|
uint16_t port = 0;
|
||
|
|
for (const auto& test : tests) {
|
||
|
|
IPEndPoint src(test.ip_address, ++port);
|
||
|
|
IPEndPoint dest(src);
|
||
|
|
EXPECT_TRUE(src == dest);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, LessThan) {
|
||
|
|
// Vary by port.
|
||
|
|
IPEndPoint ip_endpoint1(tests[0].ip_address, 100);
|
||
|
|
IPEndPoint ip_endpoint2(tests[0].ip_address, 1000);
|
||
|
|
EXPECT_TRUE(ip_endpoint1 < ip_endpoint2);
|
||
|
|
EXPECT_FALSE(ip_endpoint2 < ip_endpoint1);
|
||
|
|
|
||
|
|
// IPv4 vs IPv6
|
||
|
|
ip_endpoint1 = IPEndPoint(tests[0].ip_address, 81);
|
||
|
|
ip_endpoint2 = IPEndPoint(tests[2].ip_address, 80);
|
||
|
|
EXPECT_TRUE(ip_endpoint1 < ip_endpoint2);
|
||
|
|
EXPECT_FALSE(ip_endpoint2 < ip_endpoint1);
|
||
|
|
|
||
|
|
// IPv4 vs IPv4
|
||
|
|
ip_endpoint1 = IPEndPoint(tests[0].ip_address, 81);
|
||
|
|
ip_endpoint2 = IPEndPoint(tests[1].ip_address, 80);
|
||
|
|
EXPECT_TRUE(ip_endpoint1 < ip_endpoint2);
|
||
|
|
EXPECT_FALSE(ip_endpoint2 < ip_endpoint1);
|
||
|
|
|
||
|
|
// IPv6 vs IPv6
|
||
|
|
ip_endpoint1 = IPEndPoint(tests[2].ip_address, 81);
|
||
|
|
ip_endpoint2 = IPEndPoint(tests[3].ip_address, 80);
|
||
|
|
EXPECT_TRUE(ip_endpoint1 < ip_endpoint2);
|
||
|
|
EXPECT_FALSE(ip_endpoint2 < ip_endpoint1);
|
||
|
|
|
||
|
|
// Compare equivalent endpoints.
|
||
|
|
ip_endpoint1 = IPEndPoint(tests[0].ip_address, 80);
|
||
|
|
ip_endpoint2 = IPEndPoint(tests[0].ip_address, 80);
|
||
|
|
EXPECT_FALSE(ip_endpoint1 < ip_endpoint2);
|
||
|
|
EXPECT_FALSE(ip_endpoint2 < ip_endpoint1);
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, ToString) {
|
||
|
|
{
|
||
|
|
IPEndPoint endpoint;
|
||
|
|
EXPECT_EQ(0, endpoint.port());
|
||
|
|
}
|
||
|
|
|
||
|
|
uint16_t port = 100;
|
||
|
|
for (const auto& test : tests) {
|
||
|
|
++port;
|
||
|
|
IPEndPoint endpoint(test.ip_address, port);
|
||
|
|
const std::string result = endpoint.ToString();
|
||
|
|
EXPECT_EQ(test.host_normalized + ":" + base::NumberToString(port), result);
|
||
|
|
}
|
||
|
|
|
||
|
|
// ToString() shouldn't crash on invalid addresses.
|
||
|
|
IPAddress invalid_address;
|
||
|
|
IPEndPoint invalid_endpoint(invalid_address, 8080);
|
||
|
|
EXPECT_EQ("", invalid_endpoint.ToString());
|
||
|
|
EXPECT_EQ("", invalid_endpoint.ToStringWithoutPort());
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, RoundtripThroughValue) {
|
||
|
|
for (const auto& test : tests) {
|
||
|
|
IPEndPoint endpoint(test.ip_address, 1645);
|
||
|
|
base::Value value = endpoint.ToValue();
|
||
|
|
|
||
|
|
EXPECT_THAT(IPEndPoint::FromValue(value), Optional(endpoint));
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, FromGarbageValue) {
|
||
|
|
base::Value value(123);
|
||
|
|
EXPECT_FALSE(IPEndPoint::FromValue(value).has_value());
|
||
|
|
}
|
||
|
|
|
||
|
|
TEST_F(IPEndPointTest, FromMalformedValues) {
|
||
|
|
for (const auto& test : tests) {
|
||
|
|
base::Value valid_value = IPEndPoint(test.ip_address, 1111).ToValue();
|
||
|
|
ASSERT_TRUE(IPEndPoint::FromValue(valid_value).has_value());
|
||
|
|
|
||
|
|
base::Value missing_address = valid_value.Clone();
|
||
|
|
ASSERT_TRUE(missing_address.GetDict().Remove("address"));
|
||
|
|
EXPECT_FALSE(IPEndPoint::FromValue(missing_address).has_value());
|
||
|
|
|
||
|
|
base::Value missing_port = valid_value.Clone();
|
||
|
|
ASSERT_TRUE(missing_port.GetDict().Remove("port"));
|
||
|
|
EXPECT_FALSE(IPEndPoint::FromValue(missing_port).has_value());
|
||
|
|
|
||
|
|
base::Value invalid_address = valid_value.Clone();
|
||
|
|
*invalid_address.GetDict().Find("address") = base::Value("1.2.3.4.5");
|
||
|
|
EXPECT_FALSE(IPEndPoint::FromValue(invalid_address).has_value());
|
||
|
|
|
||
|
|
base::Value negative_port = valid_value.Clone();
|
||
|
|
*negative_port.GetDict().Find("port") = base::Value(-1);
|
||
|
|
EXPECT_FALSE(IPEndPoint::FromValue(negative_port).has_value());
|
||
|
|
|
||
|
|
base::Value large_port = valid_value.Clone();
|
||
|
|
*large_port.GetDict().Find("port") = base::Value(66000);
|
||
|
|
EXPECT_FALSE(IPEndPoint::FromValue(large_port).has_value());
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
} // namespace
|
||
|
|
|
||
|
|
} // namespace net
|