blob: acca74c5ebf0f0068563afd3923f76d3a4127bd1 [file] [log] [blame]
// Copyright 2022 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/dns/dns_names_util.h"
#include <climits>
#include <cstdint>
#include <cstring>
#include <optional>
#include <string>
#include <string_view>
#include <vector>
#include "base/containers/span_reader.h"
#include "base/numerics/safe_conversions.h"
#include "net/dns/dns_util.h"
#include "net/dns/public/dns_protocol.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace net::dns_names_util {
namespace {
using ::testing::Eq;
using ::testing::Optional;
// ToBytes converts a char* to a std::vector<uint8_t> and includes the
// terminating NUL in the result.
std::vector<uint8_t> ToBytes(const char* in) {
size_t size = strlen(in) + 1;
std::vector<uint8_t> out(size, 0);
memcpy(out.data(), in, size);
return out;
}
TEST(DnsNamesUtilTest, DottedNameToNetworkWithValidation) {
EXPECT_THAT(
DottedNameToNetwork("com", /*require_valid_internet_hostname=*/true),
Optional(ToBytes("\003com")));
EXPECT_THAT(DottedNameToNetwork("google.com",
/*require_valid_internet_hostname=*/true),
Optional(ToBytes("\x006google\003com")));
EXPECT_THAT(DottedNameToNetwork("www.google.com",
/*require_valid_internet_hostname=*/true),
Optional(ToBytes("\003www\006google\003com")));
}
TEST(DnsNamesUtilTest, DottedNameToNetwork) {
EXPECT_THAT(
DottedNameToNetwork("com", /*require_valid_internet_hostname=*/false),
Optional(ToBytes("\003com")));
EXPECT_THAT(DottedNameToNetwork("google.com",
/*require_valid_internet_hostname=*/false),
Optional(ToBytes("\x006google\003com")));
EXPECT_THAT(DottedNameToNetwork("www.google.com",
/*require_valid_internet_hostname=*/false),
Optional(ToBytes("\003www\006google\003com")));
}
TEST(DnsNamesUtilTest, DottedNameToNetworkWithValidationRejectsEmptyLabels) {
EXPECT_FALSE(DottedNameToNetwork("", /*require_valid_internet_hostname=*/true)
.has_value());
EXPECT_FALSE(
DottedNameToNetwork(".", /*require_valid_internet_hostname=*/true)
.has_value());
EXPECT_FALSE(
DottedNameToNetwork("..", /*require_valid_internet_hostname=*/true)
.has_value());
EXPECT_FALSE(DottedNameToNetwork(".google.com",
/*require_valid_internet_hostname=*/true)
.has_value());
EXPECT_FALSE(DottedNameToNetwork("www..google.com",
/*require_valid_internet_hostname=*/true)
.has_value());
}
TEST(DnsNamesUtilTest, DottedNameToNetworkRejectsEmptyLabels) {
EXPECT_FALSE(
DottedNameToNetwork("", /*require_valid_internet_hostname=*/false)
.has_value());
EXPECT_FALSE(
DottedNameToNetwork(".", /*require_valid_internet_hostname=*/false)
.has_value());
EXPECT_FALSE(
DottedNameToNetwork("..", /*require_valid_internet_hostname=*/false)
.has_value());
EXPECT_FALSE(DottedNameToNetwork(".google.com",
/*require_valid_internet_hostname=*/false)
.has_value());
EXPECT_FALSE(DottedNameToNetwork("www..google.com",
/*require_valid_internet_hostname=*/false)
.has_value());
}
TEST(DnsNamesUtilTest,
DottedNameToNetworkWithValidationAcceptsEmptyLabelAtEnd) {
EXPECT_THAT(DottedNameToNetwork("www.google.com.",
/*require_valid_internet_hostname=*/true),
Optional(ToBytes("\003www\006google\003com")));
}
TEST(DnsNamesUtilTest, DottedNameToNetworkAcceptsEmptyLabelAtEnd) {
EXPECT_THAT(DottedNameToNetwork("www.google.com.",
/*require_valid_internet_hostname=*/false),
Optional(ToBytes("\003www\006google\003com")));
}
TEST(DnsNamesUtilTest, DottedNameToNetworkWithValidationAllowsLongNames) {
// Label is 63 chars: still valid
EXPECT_THAT(
DottedNameToNetwork(
"z23456789a123456789a123456789a123456789a123456789a123456789a123",
/*require_valid_internet_hostname=*/true),
Optional(ToBytes("\077z23456789a123456789a123456789a123456789a123456"
"789a123456789a123")));
EXPECT_THAT(
DottedNameToNetwork(
"z23456789a123456789a123456789a123456789a123456789a123456789a123.",
/*require_valid_internet_hostname=*/true),
Optional(ToBytes("\077z23456789a123456789a123456789a123456789a123456"
"789a123456789a123")));
// 253 characters in the name: still valid
EXPECT_THAT(
DottedNameToNetwork(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abc",
/*require_valid_internet_hostname=*/true),
Optional(ToBytes("\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\003abc")));
// 253 characters in the name plus final dot: still valid
EXPECT_THAT(
DottedNameToNetwork(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abc.",
/*require_valid_internet_hostname=*/true),
Optional(ToBytes("\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\003abc")));
}
TEST(DnsNamesUtilTest, DottedNameToNetworkAllowsLongNames) {
// Label is 63 chars: still valid
EXPECT_THAT(
DottedNameToNetwork(
"z23456789a123456789a123456789a123456789a123456789a123456789a123",
/*require_valid_internet_hostname=*/false),
Optional(ToBytes("\077z23456789a123456789a123456789a123456789a123456"
"789a123456789a123")));
// Label is 63 chars: still valid
EXPECT_THAT(
DottedNameToNetwork(
"z23456789a123456789a123456789a123456789a123456789a123456789a123.",
/*require_valid_internet_hostname=*/false),
Optional(ToBytes("\077z23456789a123456789a123456789a123456789a123456"
"789a123456789a123")));
// 253 characters in the name: still valid
EXPECT_THAT(
DottedNameToNetwork(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abc",
/*require_valid_internet_hostname=*/false),
Optional(ToBytes("\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\003abc")));
// 253 characters in the name plus final dot: still valid
EXPECT_THAT(
DottedNameToNetwork(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abc.",
/*require_valid_internet_hostname=*/false),
Optional(ToBytes("\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\011abcdefghi\011abcdefghi\011abcdefghi"
"\011abcdefghi\003abc")));
}
TEST(DnsNamesUtilTest, DottedNameToNetworkWithValidationRejectsTooLongNames) {
// Label is too long: invalid
EXPECT_FALSE(
DottedNameToNetwork(
"123456789a123456789a123456789a123456789a123456789a123456789a1234",
/*require_valid_internet_hostname=*/true)
.has_value());
EXPECT_FALSE(
DottedNameToNetwork(
"123456789a123456789a123456789a123456789a123456789a123456789a1234.",
/*require_valid_internet_hostname=*/true)
.has_value());
// 254 characters in the name: invalid
EXPECT_FALSE(
DottedNameToNetwork(
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.1234",
/*require_valid_internet_hostname=*/true)
.has_value());
EXPECT_FALSE(
DottedNameToNetwork(
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.1234.",
/*require_valid_internet_hostname=*/true)
.has_value());
// 255 characters in the name: invalid before even trying to add a final
// zero-length termination
EXPECT_FALSE(
DottedNameToNetwork(
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.12345",
/*require_valid_internet_hostname=*/true)
.has_value());
EXPECT_FALSE(
DottedNameToNetwork(
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.12345.",
/*require_valid_internet_hostname=*/true)
.has_value());
}
TEST(DnsNamesUtilTest, DottedNameToNetworkRejectsTooLongNames) {
// Label is too long: invalid
EXPECT_FALSE(
DottedNameToNetwork(
"123456789a123456789a123456789a123456789a123456789a123456789a1234",
/*require_valid_internet_hostname=*/false)
.has_value());
EXPECT_FALSE(
DottedNameToNetwork(
"123456789a123456789a123456789a123456789a123456789a123456789a1234.",
/*require_valid_internet_hostname=*/false)
.has_value());
// 254 characters in the name: invalid
EXPECT_FALSE(
DottedNameToNetwork(
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.1234",
/*require_valid_internet_hostname=*/false)
.has_value());
EXPECT_FALSE(
DottedNameToNetwork(
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.1234.",
/*require_valid_internet_hostname=*/false)
.has_value());
// 255 characters in the name: invalid before even trying to add a final
// zero-length termination
EXPECT_FALSE(
DottedNameToNetwork(
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.12345",
/*require_valid_internet_hostname=*/false)
.has_value());
EXPECT_FALSE(
DottedNameToNetwork(
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.123456789.123456789.123456789.123456789.123456789."
"123456789.12345.",
/*require_valid_internet_hostname=*/false)
.has_value());
}
TEST(DnsNamesUtilTest,
DottedNameToNetworkWithValidationRejectsRestrictedCharacters) {
EXPECT_FALSE(DottedNameToNetwork("foo,bar.com",
/*require_valid_internet_hostname=*/true)
.has_value());
EXPECT_FALSE(DottedNameToNetwork("_ipp._tcp.local.foo printer (bar)",
/*require_valid_internet_hostname=*/true)
.has_value());
}
TEST(DnsNamesUtilTest, DottedNameToNetworkAcceptsRestrictedCharacters) {
EXPECT_THAT(DottedNameToNetwork("foo,bar.com",
/*require_valid_internet_hostname=*/false),
Optional(ToBytes("\007foo,bar\003com")));
EXPECT_THAT(
DottedNameToNetwork("_ipp._tcp.local.foo printer (bar)",
/*require_valid_internet_hostname=*/false),
Optional(ToBytes("\004_ipp\004_tcp\005local\021foo printer (bar)")));
}
TEST(DnsNamesUtilTest, NetworkToDottedNameAdvancesReader) {
{
auto reader = base::SpanReader(base::byte_span_from_cstring("\003com"));
EXPECT_THAT(NetworkToDottedName(reader), Optional(Eq("com")));
EXPECT_EQ(reader.remaining(), 0u);
}
{
auto reader =
base::SpanReader(base::byte_span_from_cstring("\003com\0ZZZ"));
EXPECT_THAT(NetworkToDottedName(reader), Optional(Eq("com")));
EXPECT_EQ(reader.remaining(), 3u);
}
}
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldHandleSimpleNames) {
std::string dns_name = "\003foo";
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name)),
Optional(Eq("foo")));
dns_name += "\003bar";
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name)),
Optional(Eq("foo.bar")));
dns_name += "\002uk";
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name)),
Optional(Eq("foo.bar.uk")));
dns_name += '\0';
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name)),
Optional(Eq("foo.bar.uk")));
}
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldHandleEmpty) {
std::string dns_name;
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name)),
Optional(Eq("")));
dns_name += '\0';
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name)),
Optional(Eq("")));
}
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldRejectEmptyIncomplete) {
std::string dns_name;
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name),
false /* require_complete */),
Optional(Eq("")));
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name),
true /* require_complete */),
std::nullopt);
}
// Test `require_complete` functionality given an input with terminating zero-
// length label.
TEST(DnsNamesUtilTest, NetworkToDottedNameComplete) {
std::string dns_name("\003foo\004test");
dns_name += '\0';
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name),
false /* require_complete */),
Optional(Eq("foo.test")));
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name),
true /* require_complete */),
Optional(Eq("foo.test")));
}
// Test `require_complete` functionality given an input without terminating
// zero-length label.
TEST(DnsNamesUtilTest, NetworkToDottedNameNotComplete) {
std::string dns_name("\003boo\004test");
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name),
false /* require_complete */),
Optional(Eq("boo.test")));
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name),
true /* require_complete */),
std::nullopt);
}
TEST(DnsNamesUtilTest,
NetworkToDottedNameShouldRejectEmptyWhenRequiringComplete) {
std::string dns_name;
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name),
false /* require_complete */),
Optional(Eq("")));
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name),
true /* require_complete */),
std::nullopt);
dns_name += '\0';
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name),
true /* require_complete */),
Optional(Eq("")));
}
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldRejectCompression) {
std::string dns_name = CreateNamePointer(152);
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
dns_name = "\005hello";
dns_name += CreateNamePointer(152);
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
}
// Test that extra input past the terminating zero-length label are ignored.
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldHandleExcessInput) {
std::string dns_name("\004cool\004name\004test");
dns_name += '\0';
dns_name += "blargh!";
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name)),
Optional(Eq("cool.name.test")));
dns_name = "\002hi";
dns_name += '\0';
dns_name += "goodbye";
EXPECT_THAT(NetworkToDottedName(base::as_byte_span(dns_name)),
Optional(Eq("hi")));
}
// Test that input is malformed if it ends mid label.
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldRejectTruncatedNames) {
std::string dns_name = "\07cheese";
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
dns_name = "\006cheesy\05test";
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
}
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldHandleLongSingleLabel) {
std::string dns_name(1, static_cast<char>(dns_protocol::kMaxLabelLength));
for (int i = 0; i < dns_protocol::kMaxLabelLength; ++i) {
dns_name += 'a';
}
EXPECT_NE(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
}
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldHandleLongSecondLabel) {
std::string dns_name("\003foo");
dns_name += static_cast<char>(dns_protocol::kMaxLabelLength);
for (int i = 0; i < dns_protocol::kMaxLabelLength; ++i) {
dns_name += 'a';
}
EXPECT_NE(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
}
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldRejectTooLongSingleLabel) {
std::string dns_name(1, static_cast<char>(dns_protocol::kMaxLabelLength));
for (int i = 0; i < dns_protocol::kMaxLabelLength + 1; ++i) {
dns_name += 'a';
}
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
}
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldRejectTooLongSecondLabel) {
std::string dns_name("\003foo");
dns_name += static_cast<char>(dns_protocol::kMaxLabelLength);
for (int i = 0; i < dns_protocol::kMaxLabelLength + 1; ++i) {
dns_name += 'a';
}
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
}
#if CHAR_MIN < 0
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldRejectCharMinLabels) {
ASSERT_GT(static_cast<uint8_t>(CHAR_MIN), dns_protocol::kMaxLabelLength);
std::string dns_name;
dns_name += base::checked_cast<char>(CHAR_MIN);
// Wherever possible, make the name otherwise valid.
if (static_cast<uint8_t>(CHAR_MIN) < UINT8_MAX) {
for (uint8_t i = 0; i < static_cast<uint8_t>(CHAR_MIN); ++i) {
dns_name += 'a';
}
}
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
}
#endif // if CHAR_MIN < 0
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldHandleLongName) {
std::string dns_name;
for (int i = 0; i < dns_protocol::kMaxNameLength;
i += (dns_protocol::kMaxLabelLength + 1)) {
int label_size = std::min(dns_protocol::kMaxNameLength - 1 - i,
dns_protocol::kMaxLabelLength);
dns_name += static_cast<char>(label_size);
for (int j = 0; j < label_size; ++j) {
dns_name += 'a';
}
}
ASSERT_EQ(dns_name.size(), static_cast<size_t>(dns_protocol::kMaxNameLength));
EXPECT_NE(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
}
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldRejectTooLongName) {
std::string dns_name;
for (int i = 0; i < dns_protocol::kMaxNameLength + 1;
i += (dns_protocol::kMaxLabelLength + 1)) {
int label_size = std::min(dns_protocol::kMaxNameLength - i,
dns_protocol::kMaxLabelLength);
dns_name += static_cast<char>(label_size);
for (int j = 0; j < label_size; ++j) {
dns_name += 'a';
}
}
ASSERT_EQ(dns_name.size(),
static_cast<size_t>(dns_protocol::kMaxNameLength + 1));
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
}
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldHandleLongCompleteName) {
std::string dns_name;
for (int i = 0; i < dns_protocol::kMaxNameLength;
i += (dns_protocol::kMaxLabelLength + 1)) {
int label_size = std::min(dns_protocol::kMaxNameLength - 1 - i,
dns_protocol::kMaxLabelLength);
dns_name += static_cast<char>(label_size);
for (int j = 0; j < label_size; ++j) {
dns_name += 'a';
}
}
dns_name += '\0';
ASSERT_EQ(dns_name.size(),
static_cast<size_t>(dns_protocol::kMaxNameLength + 1));
EXPECT_NE(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
}
TEST(DnsNamesUtilTest, NetworkToDottedNameShouldRejectTooLongCompleteName) {
std::string dns_name;
for (int i = 0; i < dns_protocol::kMaxNameLength + 1;
i += (dns_protocol::kMaxLabelLength + 1)) {
int label_size = std::min(dns_protocol::kMaxNameLength - i,
dns_protocol::kMaxLabelLength);
dns_name += static_cast<char>(label_size);
for (int j = 0; j < label_size; ++j) {
dns_name += 'a';
}
}
dns_name += '\0';
ASSERT_EQ(dns_name.size(),
static_cast<size_t>(dns_protocol::kMaxNameLength + 2));
EXPECT_EQ(NetworkToDottedName(base::as_byte_span(dns_name)), std::nullopt);
}
TEST(DnsNamesUtilTest, ValidDnsNames) {
constexpr std::string_view kGoodHostnames[] = {
"www.noodles.blorg", "1www.noodles.blorg", "www.2noodles.blorg",
"www.n--oodles.blorg", "www.noodl_es.blorg", "www.no-_odles.blorg",
"www_.noodles.blorg", "www.noodles.blorg.", "_privet._tcp.local",
"%20%20noodles.blorg", "noo dles.blorg ", "noo dles_ipp._tcp.local",
"www.nood(les).blorg", "noo dl(es)._tcp.local",
};
for (std::string_view good_hostname : kGoodHostnames) {
EXPECT_TRUE(IsValidDnsName(good_hostname));
EXPECT_TRUE(IsValidDnsRecordName(good_hostname));
}
}
TEST(DnsNamesUtilTest, EmptyNotValidDnsName) {
EXPECT_FALSE(IsValidDnsName(""));
EXPECT_FALSE(IsValidDnsRecordName(""));
}
TEST(DnsNamesUtilTest, EmptyLabelNotValidDnsName) {
EXPECT_FALSE(IsValidDnsName("www..test"));
EXPECT_FALSE(IsValidDnsName(".foo.test"));
EXPECT_FALSE(IsValidDnsRecordName("www..test"));
EXPECT_FALSE(IsValidDnsRecordName(".foo.test"));
}
TEST(DnsNameUtilTest, LongLabelsInValidDnsNames) {
EXPECT_TRUE(IsValidDnsName(
"z23456789a123456789a123456789a123456789a123456789a123456789a123"));
EXPECT_TRUE(IsValidDnsName(
"z23456789a123456789a123456789a123456789a123456789a123456789a123."));
EXPECT_TRUE(IsValidDnsRecordName(
"z23456789a123456789a123456789a123456789a123456789a123456789a123"));
EXPECT_TRUE(IsValidDnsRecordName(
"z23456789a123456789a123456789a123456789a123456789a123456789a123."));
}
TEST(DnsNameUtilTest, TooLongLabelsInInvalidDnsNames) {
EXPECT_FALSE(IsValidDnsName(
"123456789a123456789a123456789a123456789a123456789a123456789a1234"));
EXPECT_FALSE(IsValidDnsName(
"z23456789a123456789a123456789a123456789a123456789a123456789a1234."));
EXPECT_FALSE(IsValidDnsRecordName(
"z23456789a123456789a123456789a123456789a123456789a123456789a1234"));
EXPECT_FALSE(IsValidDnsRecordName(
"z23456789a123456789a123456789a123456789a123456789a123456789a1234."));
}
TEST(DnsNameUtilTest, LongValidDnsNames) {
EXPECT_TRUE(IsValidDnsName(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abc"));
EXPECT_TRUE(IsValidDnsName(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abc."));
EXPECT_TRUE(IsValidDnsRecordName(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abc"));
EXPECT_TRUE(IsValidDnsRecordName(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abc."));
}
TEST(DnsNameUtilTest, TooLongInalidDnsNames) {
EXPECT_FALSE(IsValidDnsName(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcd"));
EXPECT_FALSE(IsValidDnsName(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcd."));
EXPECT_FALSE(IsValidDnsRecordName(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcd"));
EXPECT_FALSE(IsValidDnsRecordName(
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi.abcdefghi."
"abcdefghi.abcd."));
}
TEST(DnsNameUtilTest, LocalhostNotValidDnsRecordName) {
EXPECT_TRUE(IsValidDnsName("localhost"));
EXPECT_FALSE(IsValidDnsRecordName("localhost"));
}
TEST(DnsNameUtilTest, IpAddressNotValidDnsRecordName) {
EXPECT_TRUE(IsValidDnsName("1.2.3.4"));
EXPECT_FALSE(IsValidDnsRecordName("1.2.3.4"));
EXPECT_TRUE(IsValidDnsName("[2001:4860:4860::8888]"));
EXPECT_FALSE(IsValidDnsRecordName("[2001:4860:4860::8888]"));
EXPECT_TRUE(IsValidDnsName("2001:4860:4860::8888"));
EXPECT_FALSE(IsValidDnsRecordName("2001:4860:4860::8888"));
}
TEST(DnsUtilTest, CanonicalizeNames) {
EXPECT_EQ(UrlCanonicalizeNameIfAble("GOOGLE.test"), "google.test");
EXPECT_EQ(UrlCanonicalizeNameIfAble("g{oo}gle.test"), "g{oo}gle.test");
EXPECT_EQ(UrlCanonicalizeNameIfAble("G{OO}GLE.test"), "g{oo}gle.test");
// g�gle.test
EXPECT_EQ(UrlCanonicalizeNameIfAble("g\u00FCgle.test"), "xn--ggle-0ra.test");
EXPECT_EQ(UrlCanonicalizeNameIfAble("G\u00fcGLE.test"), "xn--ggle-0ra.test");
}
TEST(DnsUtilTest, IgnoreUncanonicalizeableNames) {
EXPECT_EQ(UrlCanonicalizeNameIfAble(""), "");
// Forbidden domain code point.
// https://url.spec.whatwg.org/#forbidden-domain-code-point
EXPECT_EQ(UrlCanonicalizeNameIfAble("g<oo>gle.test"), "g<oo>gle.test");
EXPECT_EQ(UrlCanonicalizeNameIfAble("G<OO>GLE.test"), "G<OO>GLE.test");
// Invalid UTF8 character.
EXPECT_EQ(UrlCanonicalizeNameIfAble("g\x00FCgle.test"), "g\x00fcgle.test");
EXPECT_EQ(UrlCanonicalizeNameIfAble("G\x00fcGLE.test"), "G\x00fcGLE.test");
// Disallowed ASCII character.
EXPECT_EQ(UrlCanonicalizeNameIfAble("google\n.test"), "google\n.test");
EXPECT_EQ(UrlCanonicalizeNameIfAble("GOOGLE\n.test"), "GOOGLE\n.test");
}
TEST(DnsNamesUtilTest, ReadU8LengthPrefixed) {
const uint8_t kArray[] = {'b', '4', 3, 'a', 'b', 'c', 'd'};
auto reader = base::SpanReader(base::span(kArray));
EXPECT_TRUE(reader.Skip(2u));
EXPECT_EQ(reader.remaining(), 5u);
EXPECT_EQ(reader.num_read(), 2u);
base::span<const uint8_t> s;
EXPECT_TRUE(ReadU8LengthPrefixed(reader, &s));
EXPECT_EQ(s, base::span(kArray).subspan(3u, 3u));
EXPECT_EQ(reader.remaining(), 1u);
EXPECT_EQ(reader.num_read(), 6u);
}
TEST(DnsNamesUtilTest, ReadU16LengthPrefixed) {
const uint8_t kArray[] = {'b', '4', 0, 3, 'a', 'b', 'c', 'd'};
auto reader = base::SpanReader(base::span(kArray));
EXPECT_TRUE(reader.Skip(2u));
EXPECT_EQ(reader.remaining(), 6u);
EXPECT_EQ(reader.num_read(), 2u);
base::span<const uint8_t> s;
EXPECT_TRUE(ReadU16LengthPrefixed(reader, &s));
EXPECT_EQ(s, base::span(kArray).subspan(4u, 3u));
EXPECT_EQ(reader.remaining(), 1u);
EXPECT_EQ(reader.num_read(), 7u);
}
} // namespace
} // namespace net::dns_names_util