| // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "net/dns/dns_config_service_win.h" |
| |
| #include "base/check.h" |
| #include "base/memory/free_deleter.h" |
| #include "net/base/ip_address.h" |
| #include "net/base/ip_endpoint.h" |
| #include "net/dns/public/dns_protocol.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace net { |
| |
| namespace { |
| |
| TEST(DnsConfigServiceWinTest, ParseSearchList) { |
| const struct TestCase { |
| const wchar_t* input; |
| const char* output[4]; // NULL-terminated, empty if expected false |
| } cases[] = { |
| {L"chromium.org", {"chromium.org", nullptr}}, |
| {L"chromium.org,org", {"chromium.org", "org", nullptr}}, |
| // Empty suffixes terminate the list |
| {L"crbug.com,com,,org", {"crbug.com", "com", nullptr}}, |
| // IDN are converted to punycode |
| {L"\u017c\xf3\u0142ta.pi\u0119\u015b\u0107.pl,pl", |
| {"xn--ta-4ja03asj.xn--pi-wla5e0q.pl", "pl", nullptr}}, |
| // Empty search list is invalid |
| {L"", {nullptr}}, |
| {L",,", {nullptr}}, |
| }; |
| |
| for (const auto& t : cases) { |
| std::vector<std::string> actual_output, expected_output; |
| actual_output.push_back("UNSET"); |
| for (const char* const* output = t.output; *output; ++output) { |
| expected_output.push_back(*output); |
| } |
| bool result = internal::ParseSearchList(t.input, &actual_output); |
| if (!expected_output.empty()) { |
| EXPECT_TRUE(result); |
| EXPECT_EQ(expected_output, actual_output); |
| } else { |
| EXPECT_FALSE(result) << "Unexpected parse success on " << t.input; |
| } |
| } |
| } |
| |
| struct AdapterInfo { |
| IFTYPE if_type; |
| IF_OPER_STATUS oper_status; |
| const WCHAR* dns_suffix; |
| std::string dns_server_addresses[4]; // Empty string indicates end. |
| uint16_t ports[4]; |
| }; |
| |
| std::unique_ptr<IP_ADAPTER_ADDRESSES, base::FreeDeleter> CreateAdapterAddresses( |
| const AdapterInfo* infos) { |
| size_t num_adapters = 0; |
| size_t num_addresses = 0; |
| for (size_t i = 0; infos[i].if_type; ++i) { |
| ++num_adapters; |
| for (size_t j = 0; !infos[i].dns_server_addresses[j].empty(); ++j) { |
| ++num_addresses; |
| } |
| } |
| |
| size_t heap_size = num_adapters * sizeof(IP_ADAPTER_ADDRESSES) + |
| num_addresses * (sizeof(IP_ADAPTER_DNS_SERVER_ADDRESS) + |
| sizeof(struct sockaddr_storage)); |
| std::unique_ptr<IP_ADAPTER_ADDRESSES, base::FreeDeleter> heap( |
| static_cast<IP_ADAPTER_ADDRESSES*>(malloc(heap_size))); |
| CHECK(heap.get()); |
| memset(heap.get(), 0, heap_size); |
| |
| IP_ADAPTER_ADDRESSES* adapters = heap.get(); |
| IP_ADAPTER_DNS_SERVER_ADDRESS* addresses = |
| reinterpret_cast<IP_ADAPTER_DNS_SERVER_ADDRESS*>(adapters + num_adapters); |
| struct sockaddr_storage* storage = |
| reinterpret_cast<struct sockaddr_storage*>(addresses + num_addresses); |
| |
| for (size_t i = 0; i < num_adapters; ++i) { |
| const AdapterInfo& info = infos[i]; |
| IP_ADAPTER_ADDRESSES* adapter = adapters + i; |
| if (i + 1 < num_adapters) |
| adapter->Next = adapter + 1; |
| adapter->IfType = info.if_type; |
| adapter->OperStatus = info.oper_status; |
| adapter->DnsSuffix = const_cast<PWCHAR>(info.dns_suffix); |
| IP_ADAPTER_DNS_SERVER_ADDRESS* address = nullptr; |
| for (size_t j = 0; !info.dns_server_addresses[j].empty(); ++j) { |
| --num_addresses; |
| if (j == 0) { |
| address = adapter->FirstDnsServerAddress = addresses + num_addresses; |
| } else { |
| // Note that |address| is moving backwards. |
| address = address->Next = address - 1; |
| } |
| IPAddress ip; |
| CHECK(ip.AssignFromIPLiteral(info.dns_server_addresses[j])); |
| IPEndPoint ipe = IPEndPoint(ip, info.ports[j]); |
| address->Address.lpSockaddr = |
| reinterpret_cast<LPSOCKADDR>(storage + num_addresses); |
| socklen_t length = sizeof(struct sockaddr_storage); |
| CHECK(ipe.ToSockAddr(address->Address.lpSockaddr, &length)); |
| address->Address.iSockaddrLength = static_cast<int>(length); |
| } |
| } |
| |
| return heap; |
| } |
| |
| TEST(DnsConfigServiceWinTest, ConvertAdapterAddresses) { |
| // Check nameservers and connection-specific suffix. |
| const struct TestCase { |
| AdapterInfo input_adapters[4]; // |if_type| == 0 indicates end. |
| std::string expected_nameservers[4]; // Empty string indicates end. |
| std::string expected_suffix; |
| uint16_t expected_ports[4]; |
| } cases[] = { |
| { // Ignore loopback and inactive adapters. |
| { |
| { IF_TYPE_SOFTWARE_LOOPBACK, IfOperStatusUp, L"funnyloop", |
| { "2.0.0.2" } }, |
| { IF_TYPE_FASTETHER, IfOperStatusDormant, L"example.com", |
| { "1.0.0.1" } }, |
| { IF_TYPE_USB, IfOperStatusUp, L"chromium.org", |
| { "10.0.0.10", "2001:FFFF::1111" } }, |
| { 0 }, |
| }, |
| { "10.0.0.10", "2001:FFFF::1111" }, |
| "chromium.org", |
| }, |
| { // Respect configured ports. |
| { |
| { IF_TYPE_USB, IfOperStatusUp, L"chromium.org", |
| { "10.0.0.10", "2001:FFFF::1111" }, { 1024, 24 } }, |
| { 0 }, |
| }, |
| { "10.0.0.10", "2001:FFFF::1111" }, |
| "chromium.org", |
| { 1024, 24 }, |
| }, |
| { // Use the preferred adapter (first in binding order) and filter |
| // stateless DNS discovery addresses. |
| { |
| { IF_TYPE_SOFTWARE_LOOPBACK, IfOperStatusUp, L"funnyloop", |
| { "2.0.0.2" } }, |
| { IF_TYPE_FASTETHER, IfOperStatusUp, L"example.com", |
| { "1.0.0.1", "fec0:0:0:ffff::2", "8.8.8.8" } }, |
| { IF_TYPE_USB, IfOperStatusUp, L"chromium.org", |
| { "10.0.0.10", "2001:FFFF::1111" } }, |
| { 0 }, |
| }, |
| { "1.0.0.1", "8.8.8.8" }, |
| "example.com", |
| }, |
| { // No usable adapters. |
| { |
| { IF_TYPE_SOFTWARE_LOOPBACK, IfOperStatusUp, L"localhost", |
| { "2.0.0.2" } }, |
| { IF_TYPE_FASTETHER, IfOperStatusDormant, L"example.com", |
| { "1.0.0.1" } }, |
| { IF_TYPE_USB, IfOperStatusUp, L"chromium.org" }, |
| { 0 }, |
| }, |
| }, |
| }; |
| |
| for (const auto& t : cases) { |
| internal::DnsSystemSettings settings; |
| settings.addresses = CreateAdapterAddresses(t.input_adapters); |
| // Default settings for the rest. |
| std::vector<IPEndPoint> expected_nameservers; |
| for (size_t j = 0; !t.expected_nameservers[j].empty(); ++j) { |
| IPAddress ip; |
| ASSERT_TRUE(ip.AssignFromIPLiteral(t.expected_nameservers[j])); |
| uint16_t port = t.expected_ports[j]; |
| if (!port) |
| port = dns_protocol::kDefaultPort; |
| expected_nameservers.push_back(IPEndPoint(ip, port)); |
| } |
| |
| DnsConfig config; |
| internal::ConfigParseWinResult result = |
| internal::ConvertSettingsToDnsConfig(settings, &config); |
| internal::ConfigParseWinResult expected_result = |
| expected_nameservers.empty() ? internal::CONFIG_PARSE_WIN_NO_NAMESERVERS |
| : internal::CONFIG_PARSE_WIN_OK; |
| EXPECT_EQ(expected_result, result); |
| EXPECT_EQ(expected_nameservers, config.nameservers); |
| if (result == internal::CONFIG_PARSE_WIN_OK) { |
| ASSERT_EQ(1u, config.search.size()); |
| EXPECT_EQ(t.expected_suffix, config.search[0]); |
| } |
| } |
| } |
| |
| TEST(DnsConfigServiceWinTest, ConvertSuffixSearch) { |
| AdapterInfo infos[2] = { |
| { IF_TYPE_USB, IfOperStatusUp, L"connection.suffix", { "1.0.0.1" } }, |
| { 0 }, |
| }; |
| |
| const struct TestCase { |
| struct { |
| internal::DnsSystemSettings::RegString policy_search_list; |
| internal::DnsSystemSettings::RegString tcpip_search_list; |
| internal::DnsSystemSettings::RegString tcpip_domain; |
| internal::DnsSystemSettings::RegString primary_dns_suffix; |
| internal::DnsSystemSettings::DevolutionSetting policy_devolution; |
| internal::DnsSystemSettings::DevolutionSetting dnscache_devolution; |
| internal::DnsSystemSettings::DevolutionSetting tcpip_devolution; |
| } input_settings; |
| std::string expected_search[5]; |
| } cases[] = { |
| { |
| // Policy SearchList override. |
| { |
| {true, L"policy.searchlist.a,policy.searchlist.b"}, |
| {true, L"tcpip.searchlist.a,tcpip.searchlist.b"}, |
| {true, L"tcpip.domain"}, |
| {true, L"primary.dns.suffix"}, |
| }, |
| {"policy.searchlist.a", "policy.searchlist.b"}, |
| }, |
| { |
| // User-specified SearchList override. |
| { |
| {false}, |
| {true, L"tcpip.searchlist.a,tcpip.searchlist.b"}, |
| {true, L"tcpip.domain"}, |
| {true, L"primary.dns.suffix"}, |
| }, |
| {"tcpip.searchlist.a", "tcpip.searchlist.b"}, |
| }, |
| { |
| // Void SearchList. Using tcpip.domain |
| { |
| {true, L",bad.searchlist,parsed.as.empty"}, |
| {true, L"tcpip.searchlist,good.but.overridden"}, |
| {true, L"tcpip.domain"}, |
| {false}, |
| }, |
| {"tcpip.domain", "connection.suffix"}, |
| }, |
| { |
| // Void SearchList. Using primary.dns.suffix |
| { |
| {true, L",bad.searchlist,parsed.as.empty"}, |
| {true, L"tcpip.searchlist,good.but.overridden"}, |
| {true, L"tcpip.domain"}, |
| {true, L"primary.dns.suffix"}, |
| }, |
| {"primary.dns.suffix", "connection.suffix"}, |
| }, |
| { |
| // Void SearchList. Using tcpip.domain when primary.dns.suffix is |
| // empty |
| { |
| {true, L",bad.searchlist,parsed.as.empty"}, |
| {true, L"tcpip.searchlist,good.but.overridden"}, |
| {true, L"tcpip.domain"}, |
| {true, L""}, |
| }, |
| {"tcpip.domain", "connection.suffix"}, |
| }, |
| { |
| // Void SearchList. Using tcpip.domain when primary.dns.suffix is NULL |
| { |
| {true, L",bad.searchlist,parsed.as.empty"}, |
| {true, L"tcpip.searchlist,good.but.overridden"}, |
| {true, L"tcpip.domain"}, |
| {true}, |
| }, |
| {"tcpip.domain", "connection.suffix"}, |
| }, |
| { |
| // No primary suffix. Devolution does not matter. |
| { |
| {false}, |
| {false}, |
| {true}, |
| {true}, |
| {{true, 1}, {true, 2}}, |
| }, |
| {"connection.suffix"}, |
| }, |
| { |
| // Devolution enabled by policy, level by dnscache. |
| { |
| {false}, |
| {false}, |
| {true, L"a.b.c.d.e"}, |
| {false}, |
| {{true, 1}, {false}}, // policy_devolution: enabled, level |
| {{true, 0}, {true, 3}}, // dnscache_devolution |
| {{true, 0}, {true, 1}}, // tcpip_devolution |
| }, |
| {"a.b.c.d.e", "connection.suffix", "b.c.d.e", "c.d.e"}, |
| }, |
| { |
| // Devolution enabled by dnscache, level by policy. |
| { |
| {false}, |
| {false}, |
| {true, L"a.b.c.d.e"}, |
| {true, L"f.g.i.l.j"}, |
| {{false}, {true, 4}}, |
| {{true, 1}, {false}}, |
| {{true, 0}, {true, 3}}, |
| }, |
| {"f.g.i.l.j", "connection.suffix", "g.i.l.j"}, |
| }, |
| { |
| // Devolution enabled by default. |
| { |
| {false}, |
| {false}, |
| {true, L"a.b.c.d.e"}, |
| {false}, |
| {{false}, {false}}, |
| {{false}, {true, 3}}, |
| {{false}, {true, 1}}, |
| }, |
| {"a.b.c.d.e", "connection.suffix", "b.c.d.e", "c.d.e"}, |
| }, |
| { |
| // Devolution enabled at level = 2, but nothing to devolve. |
| { |
| {false}, |
| {false}, |
| {true, L"a.b"}, |
| {false}, |
| {{false}, {false}}, |
| {{false}, {true, 2}}, |
| {{false}, {true, 2}}, |
| }, |
| {"a.b", "connection.suffix"}, |
| }, |
| { |
| // Devolution disabled when no explicit level. |
| { |
| {false}, |
| {false}, |
| {true, L"a.b.c.d.e"}, |
| {false}, |
| {{true, 1}, {false}}, |
| {{true, 1}, {false}}, |
| {{true, 1}, {false}}, |
| }, |
| {"a.b.c.d.e", "connection.suffix"}, |
| }, |
| { |
| // Devolution disabled by policy level. |
| { |
| {false}, |
| {false}, |
| {true, L"a.b.c.d.e"}, |
| {false}, |
| {{false}, {true, 1}}, |
| {{true, 1}, {true, 3}}, |
| {{true, 1}, {true, 4}}, |
| }, |
| {"a.b.c.d.e", "connection.suffix"}, |
| }, |
| { |
| // Devolution disabled by user setting. |
| { |
| {false}, |
| {false}, |
| {true, L"a.b.c.d.e"}, |
| {false}, |
| {{false}, {true, 3}}, |
| {{false}, {true, 3}}, |
| {{true, 0}, {true, 3}}, |
| }, |
| {"a.b.c.d.e", "connection.suffix"}, |
| }, |
| }; |
| |
| for (auto& t : cases) { |
| internal::DnsSystemSettings settings; |
| settings.addresses = CreateAdapterAddresses(infos); |
| settings.policy_search_list = t.input_settings.policy_search_list; |
| settings.tcpip_search_list = t.input_settings.tcpip_search_list; |
| settings.tcpip_domain = t.input_settings.tcpip_domain; |
| settings.primary_dns_suffix = t.input_settings.primary_dns_suffix; |
| settings.policy_devolution = t.input_settings.policy_devolution; |
| settings.dnscache_devolution = t.input_settings.dnscache_devolution; |
| settings.tcpip_devolution = t.input_settings.tcpip_devolution; |
| |
| DnsConfig config; |
| EXPECT_EQ(internal::CONFIG_PARSE_WIN_OK, |
| internal::ConvertSettingsToDnsConfig(settings, &config)); |
| std::vector<std::string> expected_search; |
| for (size_t j = 0; !t.expected_search[j].empty(); ++j) { |
| expected_search.push_back(t.expected_search[j]); |
| } |
| EXPECT_EQ(expected_search, config.search); |
| } |
| } |
| |
| TEST(DnsConfigServiceWinTest, AppendToMultiLabelName) { |
| AdapterInfo infos[2] = { |
| { IF_TYPE_USB, IfOperStatusUp, L"connection.suffix", { "1.0.0.1" } }, |
| { 0 }, |
| }; |
| |
| const struct TestCase { |
| internal::DnsSystemSettings::RegDword input; |
| bool expected_output; |
| } cases[] = { |
| {{true, 0}, false}, {{true, 1}, true}, {{false, 0}, false}, |
| }; |
| |
| for (const auto& t : cases) { |
| internal::DnsSystemSettings settings; |
| settings.addresses = CreateAdapterAddresses(infos); |
| settings.append_to_multi_label_name = t.input; |
| DnsConfig config; |
| EXPECT_EQ(internal::CONFIG_PARSE_WIN_OK, |
| internal::ConvertSettingsToDnsConfig(settings, &config)); |
| EXPECT_EQ(t.expected_output, config.append_to_multi_label_name); |
| } |
| } |
| |
| // Setting have_name_resolution_policy_table should set unhandled_options. |
| TEST(DnsConfigServiceWinTest, HaveNRPT) { |
| AdapterInfo infos[2] = { |
| { IF_TYPE_USB, IfOperStatusUp, L"connection.suffix", { "1.0.0.1" } }, |
| { 0 }, |
| }; |
| |
| const struct TestCase { |
| bool have_nrpt; |
| bool unhandled_options; |
| internal::ConfigParseWinResult result; |
| } cases[] = { |
| { false, false, internal::CONFIG_PARSE_WIN_OK }, |
| { true, true, internal::CONFIG_PARSE_WIN_UNHANDLED_OPTIONS }, |
| }; |
| |
| for (const auto& t : cases) { |
| internal::DnsSystemSettings settings; |
| settings.addresses = CreateAdapterAddresses(infos); |
| settings.have_name_resolution_policy = t.have_nrpt; |
| DnsConfig config; |
| EXPECT_EQ(t.result, |
| internal::ConvertSettingsToDnsConfig(settings, &config)); |
| EXPECT_EQ(t.unhandled_options, config.unhandled_options); |
| EXPECT_EQ(t.have_nrpt, config.use_local_ipv6); |
| } |
| } |
| |
| // Setting have_proxy should set unhandled_options. |
| TEST(DnsConfigServiceWinTest, HaveProxy) { |
| AdapterInfo infos[2] = { |
| {IF_TYPE_USB, IfOperStatusUp, L"connection.suffix", {"1.0.0.1"}}, |
| {0}, |
| }; |
| |
| const struct TestCase { |
| bool have_proxy; |
| bool unhandled_options; |
| internal::ConfigParseWinResult result; |
| } cases[] = { |
| {false, false, internal::CONFIG_PARSE_WIN_OK}, |
| {true, true, internal::CONFIG_PARSE_WIN_UNHANDLED_OPTIONS}, |
| }; |
| |
| for (const auto& t : cases) { |
| internal::DnsSystemSettings settings; |
| settings.addresses = CreateAdapterAddresses(infos); |
| settings.have_proxy = t.have_proxy; |
| DnsConfig config; |
| EXPECT_EQ(t.result, |
| internal::ConvertSettingsToDnsConfig(settings, &config)); |
| EXPECT_EQ(t.unhandled_options, config.unhandled_options); |
| } |
| } |
| |
| // Setting uses_vpn should set unhandled_options. |
| TEST(DnsConfigServiceWinTest, UsesVpn) { |
| AdapterInfo infos[3] = { |
| {IF_TYPE_USB, IfOperStatusUp, L"connection.suffix", {"1.0.0.1"}}, |
| {IF_TYPE_PPP, IfOperStatusUp, L"connection.suffix", {"1.0.0.1"}}, |
| {0}, |
| }; |
| |
| internal::DnsSystemSettings settings; |
| settings.addresses = CreateAdapterAddresses(infos); |
| DnsConfig config; |
| EXPECT_EQ(internal::CONFIG_PARSE_WIN_UNHANDLED_OPTIONS, |
| internal::ConvertSettingsToDnsConfig(settings, &config)); |
| EXPECT_TRUE(config.unhandled_options); |
| } |
| |
| } // namespace |
| |
| } // namespace net |