blob: 024115ad7602df661af9f12207044ec90c7aabea [file] [log] [blame]
henrike@webrtc.orgf0488722014-05-13 18:00:26 +00001/*
2 * Copyright 2004 The WebRTC Project Authors. All rights reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "rtc_base/network.h"
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000012
oprypin1ea631f2017-08-18 00:15:19 -070013#include <stdlib.h>
14
jbauch555604a2016-04-26 03:13:22 -070015#include <memory>
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000016#include <vector>
oprypin1ea631f2017-08-18 00:15:19 -070017
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020018#include "rtc_base/checks.h"
Steve Anton10542f22019-01-11 09:11:00 -080019#include "rtc_base/net_helpers.h"
20#include "rtc_base/network_monitor.h"
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000021#if defined(WEBRTC_POSIX)
Henrik Kjellander00725112017-06-30 15:14:45 +020022#include <net/if.h>
Yves Gerey665174f2018-06-19 15:03:05 +020023#include <sys/types.h>
Jonas Olssona4d87372019-07-05 19:08:33 +020024
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020025#include "rtc_base/ifaddrs_converter.h"
Guo-wei Shieh9faf1542015-12-28 14:06:55 -080026#endif // defined(WEBRTC_POSIX)
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020027#include "rtc_base/gunit.h"
Steve Anton2acd1632019-03-25 13:48:30 -070028#include "test/gmock.h"
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000029#if defined(WEBRTC_WIN)
Mirko Bonadei675513b2017-11-09 11:09:25 +010030#include "rtc_base/logging.h" // For RTC_LOG_GLE
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000031#endif
32
Steve Anton2acd1632019-03-25 13:48:30 -070033using ::testing::Contains;
34using ::testing::Not;
35using ::testing::UnorderedElementsAre;
36using ::testing::UnorderedElementsAreArray;
37
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000038namespace rtc {
39
Guo-wei Shieh9faf1542015-12-28 14:06:55 -080040namespace {
41
honghaiz023f3ef2015-10-19 09:39:32 -070042class FakeNetworkMonitor : public NetworkMonitorBase {
43 public:
honghaizcec0a082016-01-15 14:49:09 -080044 void Start() override { started_ = true; }
45 void Stop() override { started_ = false; }
46 bool started() { return started_; }
honghaiza7ad7c32016-02-02 12:54:14 -080047 AdapterType GetAdapterType(const std::string& if_name) override {
Honghai Zhang351d77b2016-05-20 15:08:29 -070048 // Note that the name matching rules are different from the
49 // GetAdapterTypeFromName in NetworkManager.
50 if (if_name.find("wifi") == 0) {
51 return ADAPTER_TYPE_WIFI;
52 }
53 if (if_name.find("cellular") == 0) {
54 return ADAPTER_TYPE_CELLULAR;
55 }
honghaiza7ad7c32016-02-02 12:54:14 -080056 return ADAPTER_TYPE_UNKNOWN;
57 }
honghaizcec0a082016-01-15 14:49:09 -080058
59 private:
60 bool started_ = false;
honghaiz023f3ef2015-10-19 09:39:32 -070061};
62
63class FakeNetworkMonitorFactory : public NetworkMonitorFactory {
64 public:
65 FakeNetworkMonitorFactory() {}
honghaizcec0a082016-01-15 14:49:09 -080066 NetworkMonitorInterface* CreateNetworkMonitor() override {
honghaiz023f3ef2015-10-19 09:39:32 -070067 return new FakeNetworkMonitor();
68 }
69};
70
Qingsi Wang10a0e512018-05-16 13:37:03 -070071bool SameNameAndPrefix(const rtc::Network& a, const rtc::Network& b) {
72 if (a.name() != b.name()) {
73 RTC_LOG(INFO) << "Different interface names.";
74 return false;
75 }
76 if (a.prefix_length() != b.prefix_length() || a.prefix() != b.prefix()) {
77 RTC_LOG(INFO) << "Different IP prefixes.";
78 return false;
79 }
80 return true;
81}
82
Guo-wei Shieh9faf1542015-12-28 14:06:55 -080083} // namespace
84
Mirko Bonadei6a489f22019-04-09 15:11:12 +020085class NetworkTest : public ::testing::Test, public sigslot::has_slots<> {
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000086 public:
87 NetworkTest() : callback_called_(false) {}
88
Yves Gerey665174f2018-06-19 15:03:05 +020089 void OnNetworksChanged() { callback_called_ = true; }
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000090
guoweis@webrtc.orga094cac2015-01-28 19:34:05 +000091 NetworkManager::Stats MergeNetworkList(
92 BasicNetworkManager& network_manager,
93 const NetworkManager::NetworkList& list,
94 bool* changed) {
95 NetworkManager::Stats stats;
96 network_manager.MergeNetworkList(list, changed, &stats);
97 return stats;
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000098 }
99
100 bool IsIgnoredNetwork(BasicNetworkManager& network_manager,
101 const Network& network) {
102 return network_manager.IsIgnoredNetwork(network);
103 }
104
105 NetworkManager::NetworkList GetNetworks(
Yves Gerey665174f2018-06-19 15:03:05 +0200106 const BasicNetworkManager& network_manager,
107 bool include_ignored) {
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000108 NetworkManager::NetworkList list;
109 network_manager.CreateNetworks(include_ignored, &list);
110 return list;
111 }
112
honghaizcec0a082016-01-15 14:49:09 -0800113 FakeNetworkMonitor* GetNetworkMonitor(BasicNetworkManager& network_manager) {
114 return static_cast<FakeNetworkMonitor*>(
115 network_manager.network_monitor_.get());
honghaiz023f3ef2015-10-19 09:39:32 -0700116 }
117 void ClearNetworks(BasicNetworkManager& network_manager) {
118 for (const auto& kv : network_manager.networks_map_) {
119 delete kv.second;
120 }
121 network_manager.networks_.clear();
122 network_manager.networks_map_.clear();
123 }
124
Honghai Zhang351d77b2016-05-20 15:08:29 -0700125 AdapterType GetAdapterType(BasicNetworkManager& network_manager) {
126 BasicNetworkManager::NetworkList list;
127 network_manager.GetNetworks(&list);
nissec16fa5e2017-02-07 07:18:43 -0800128 RTC_CHECK_EQ(1, list.size());
Honghai Zhang351d77b2016-05-20 15:08:29 -0700129 return list[0]->type();
130 }
131
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000132#if defined(WEBRTC_POSIX)
133 // Separated from CreateNetworks for tests.
134 static void CallConvertIfAddrs(const BasicNetworkManager& network_manager,
135 struct ifaddrs* interfaces,
136 bool include_ignored,
137 NetworkManager::NetworkList* networks) {
Guo-wei Shieh9faf1542015-12-28 14:06:55 -0800138 // Use the base IfAddrsConverter for test cases.
jbauch555604a2016-04-26 03:13:22 -0700139 std::unique_ptr<IfAddrsConverter> ifaddrs_converter(new IfAddrsConverter());
Guo-wei Shieh9faf1542015-12-28 14:06:55 -0800140 network_manager.ConvertIfAddrs(interfaces, ifaddrs_converter.get(),
141 include_ignored, networks);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000142 }
honghaizdb8cf502015-12-21 13:08:46 -0800143
144 struct sockaddr_in6* CreateIpv6Addr(const std::string& ip_string,
145 uint32_t scope_id) {
Yves Gerey665174f2018-06-19 15:03:05 +0200146 struct sockaddr_in6* ipv6_addr =
147 static_cast<struct sockaddr_in6*>(malloc(sizeof(struct sockaddr_in6)));
honghaizdb8cf502015-12-21 13:08:46 -0800148 memset(ipv6_addr, 0, sizeof(struct sockaddr_in6));
149 ipv6_addr->sin6_family = AF_INET6;
150 ipv6_addr->sin6_scope_id = scope_id;
151 IPAddress ip;
152 IPFromString(ip_string, &ip);
153 ipv6_addr->sin6_addr = ip.ipv6_address();
154 return ipv6_addr;
155 }
156
157 // Pointers created here need to be released via ReleaseIfAddrs.
158 struct ifaddrs* AddIpv6Address(struct ifaddrs* list,
159 char* if_name,
160 const std::string& ipv6_address,
161 const std::string& ipv6_netmask,
162 uint32_t scope_id) {
163 struct ifaddrs* if_addr = new struct ifaddrs;
164 memset(if_addr, 0, sizeof(struct ifaddrs));
165 if_addr->ifa_name = if_name;
166 if_addr->ifa_addr = reinterpret_cast<struct sockaddr*>(
167 CreateIpv6Addr(ipv6_address, scope_id));
168 if_addr->ifa_netmask =
169 reinterpret_cast<struct sockaddr*>(CreateIpv6Addr(ipv6_netmask, 0));
170 if_addr->ifa_next = list;
Guo-wei Shieh9faf1542015-12-28 14:06:55 -0800171 if_addr->ifa_flags = IFF_RUNNING;
honghaizdb8cf502015-12-21 13:08:46 -0800172 return if_addr;
173 }
174
Honghai Zhang351d77b2016-05-20 15:08:29 -0700175 struct ifaddrs* InstallIpv6Network(char* if_name,
176 const std::string& ipv6_address,
177 const std::string& ipv6_mask,
178 BasicNetworkManager& network_manager) {
179 ifaddrs* addr_list = nullptr;
180 addr_list = AddIpv6Address(addr_list, if_name, ipv6_address, ipv6_mask, 0);
181 NetworkManager::NetworkList result;
182 bool changed;
183 NetworkManager::Stats stats;
184 CallConvertIfAddrs(network_manager, addr_list, true, &result);
185 network_manager.MergeNetworkList(result, &changed, &stats);
186 return addr_list;
187 }
188
Jeroen de Borst8f096d02019-02-21 13:34:45 -0800189 struct sockaddr_in* CreateIpv4Addr(const std::string& ip_string) {
190 struct sockaddr_in* ipv4_addr =
191 static_cast<struct sockaddr_in*>(malloc(sizeof(struct sockaddr_in)));
192 memset(ipv4_addr, 0, sizeof(struct sockaddr_in));
193 ipv4_addr->sin_family = AF_INET;
194 IPAddress ip;
195 IPFromString(ip_string, &ip);
196 ipv4_addr->sin_addr = ip.ipv4_address();
197 return ipv4_addr;
198 }
199
200 // Pointers created here need to be released via ReleaseIfAddrs.
201 struct ifaddrs* AddIpv4Address(struct ifaddrs* list,
202 char* if_name,
203 const std::string& ipv4_address,
204 const std::string& ipv4_netmask) {
205 struct ifaddrs* if_addr = new struct ifaddrs;
206 memset(if_addr, 0, sizeof(struct ifaddrs));
207 if_addr->ifa_name = if_name;
208 if_addr->ifa_addr =
209 reinterpret_cast<struct sockaddr*>(CreateIpv4Addr(ipv4_address));
210 if_addr->ifa_netmask =
211 reinterpret_cast<struct sockaddr*>(CreateIpv4Addr(ipv4_netmask));
212 if_addr->ifa_next = list;
213 if_addr->ifa_flags = IFF_RUNNING;
214 return if_addr;
215 }
216
217 struct ifaddrs* InstallIpv4Network(char* if_name,
218 const std::string& ipv4_address,
219 const std::string& ipv4_mask,
220 BasicNetworkManager& network_manager) {
221 ifaddrs* addr_list = nullptr;
222 addr_list = AddIpv4Address(addr_list, if_name, ipv4_address, ipv4_mask);
223 NetworkManager::NetworkList result;
224 bool changed;
225 NetworkManager::Stats stats;
226 CallConvertIfAddrs(network_manager, addr_list, true, &result);
227 network_manager.MergeNetworkList(result, &changed, &stats);
228 return addr_list;
229 }
230
honghaizdb8cf502015-12-21 13:08:46 -0800231 void ReleaseIfAddrs(struct ifaddrs* list) {
232 struct ifaddrs* if_addr = list;
233 while (if_addr != nullptr) {
234 struct ifaddrs* next_addr = if_addr->ifa_next;
oprypin1ea631f2017-08-18 00:15:19 -0700235 free(if_addr->ifa_addr);
236 free(if_addr->ifa_netmask);
honghaizdb8cf502015-12-21 13:08:46 -0800237 delete if_addr;
238 if_addr = next_addr;
239 }
240 }
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000241#endif // defined(WEBRTC_POSIX)
242
243 protected:
244 bool callback_called_;
245};
246
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800247class TestBasicNetworkManager : public BasicNetworkManager {
248 public:
249 using BasicNetworkManager::QueryDefaultLocalAddress;
Guo-wei Shieha34c39e2015-11-25 13:12:26 -0800250 using BasicNetworkManager::set_default_local_addresses;
Guo-wei Shieh9af97f82015-11-10 14:47:39 -0800251};
252
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000253// Test that the Network ctor works properly.
254TEST_F(NetworkTest, TestNetworkConstruct) {
255 Network ipv4_network1("test_eth0", "Test Network Adapter 1",
256 IPAddress(0x12345600U), 24);
257 EXPECT_EQ("test_eth0", ipv4_network1.name());
258 EXPECT_EQ("Test Network Adapter 1", ipv4_network1.description());
259 EXPECT_EQ(IPAddress(0x12345600U), ipv4_network1.prefix());
260 EXPECT_EQ(24, ipv4_network1.prefix_length());
261 EXPECT_FALSE(ipv4_network1.ignored());
262}
263
phoglund@webrtc.org006521d2015-02-12 09:23:59 +0000264TEST_F(NetworkTest, TestIsIgnoredNetworkIgnoresIPsStartingWith0) {
265 Network ipv4_network1("test_eth0", "Test Network Adapter 1",
266 IPAddress(0x12345600U), 24, ADAPTER_TYPE_ETHERNET);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000267 Network ipv4_network2("test_eth1", "Test Network Adapter 2",
phoglund@webrtc.org006521d2015-02-12 09:23:59 +0000268 IPAddress(0x010000U), 24, ADAPTER_TYPE_ETHERNET);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000269 BasicNetworkManager network_manager;
270 EXPECT_FALSE(IsIgnoredNetwork(network_manager, ipv4_network1));
271 EXPECT_TRUE(IsIgnoredNetwork(network_manager, ipv4_network2));
272}
273
phoglund@webrtc.org006521d2015-02-12 09:23:59 +0000274// TODO(phoglund): Remove when ignore list goes away.
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000275TEST_F(NetworkTest, TestIgnoreList) {
Yves Gerey665174f2018-06-19 15:03:05 +0200276 Network ignore_me("ignore_me", "Ignore me please!", IPAddress(0x12345600U),
277 24);
278 Network include_me("include_me", "Include me please!", IPAddress(0x12345600U),
279 24);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000280 BasicNetworkManager network_manager;
281 EXPECT_FALSE(IsIgnoredNetwork(network_manager, ignore_me));
282 EXPECT_FALSE(IsIgnoredNetwork(network_manager, include_me));
283 std::vector<std::string> ignore_list;
284 ignore_list.push_back("ignore_me");
285 network_manager.set_network_ignore_list(ignore_list);
286 EXPECT_TRUE(IsIgnoredNetwork(network_manager, ignore_me));
287 EXPECT_FALSE(IsIgnoredNetwork(network_manager, include_me));
288}
289
290// Test is failing on Windows opt: b/11288214
291TEST_F(NetworkTest, DISABLED_TestCreateNetworks) {
292 BasicNetworkManager manager;
293 NetworkManager::NetworkList result = GetNetworks(manager, true);
294 // We should be able to bind to any addresses we find.
295 NetworkManager::NetworkList::iterator it;
Yves Gerey665174f2018-06-19 15:03:05 +0200296 for (it = result.begin(); it != result.end(); ++it) {
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000297 sockaddr_storage storage;
298 memset(&storage, 0, sizeof(storage));
guoweis@webrtc.org369a6372014-09-17 22:37:29 +0000299 IPAddress ip = (*it)->GetBestIP();
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000300 SocketAddress bindaddress(ip, 0);
301 bindaddress.SetScopeID((*it)->scope_id());
302 // TODO(thaloun): Use rtc::AsyncSocket once it supports IPv6.
303 int fd = static_cast<int>(socket(ip.family(), SOCK_STREAM, IPPROTO_TCP));
304 if (fd > 0) {
305 size_t ipsize = bindaddress.ToSockAddrStorage(&storage);
306 EXPECT_GE(ipsize, 0U);
Yves Gerey665174f2018-06-19 15:03:05 +0200307 int success = ::bind(fd, reinterpret_cast<sockaddr*>(&storage),
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000308 static_cast<int>(ipsize));
309#if defined(WEBRTC_WIN)
Mirko Bonadei675513b2017-11-09 11:09:25 +0100310 if (success)
311 RTC_LOG_GLE(LS_ERROR) << "Socket bind failed.";
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000312#endif
313 EXPECT_EQ(0, success);
314#if defined(WEBRTC_WIN)
315 closesocket(fd);
316#else
317 close(fd);
318#endif
319 }
320 delete (*it);
321 }
322}
323
Guo-wei Shieh47872ec2015-08-19 10:32:46 -0700324// Test StartUpdating() and StopUpdating(). network_permission_state starts with
325// ALLOWED.
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000326TEST_F(NetworkTest, TestUpdateNetworks) {
327 BasicNetworkManager manager;
Yves Gerey665174f2018-06-19 15:03:05 +0200328 manager.SignalNetworksChanged.connect(static_cast<NetworkTest*>(this),
329 &NetworkTest::OnNetworksChanged);
guoweisea1012b2015-08-21 09:06:28 -0700330 EXPECT_EQ(NetworkManager::ENUMERATION_ALLOWED,
331 manager.enumeration_permission());
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000332 manager.StartUpdating();
333 Thread::Current()->ProcessMessages(0);
334 EXPECT_TRUE(callback_called_);
335 callback_called_ = false;
336 // Callback should be triggered immediately when StartUpdating
337 // is called, after network update signal is already sent.
338 manager.StartUpdating();
339 EXPECT_TRUE(manager.started());
340 Thread::Current()->ProcessMessages(0);
341 EXPECT_TRUE(callback_called_);
342 manager.StopUpdating();
343 EXPECT_TRUE(manager.started());
344 manager.StopUpdating();
guoweisea1012b2015-08-21 09:06:28 -0700345 EXPECT_EQ(NetworkManager::ENUMERATION_ALLOWED,
346 manager.enumeration_permission());
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000347 EXPECT_FALSE(manager.started());
348 manager.StopUpdating();
349 EXPECT_FALSE(manager.started());
350 callback_called_ = false;
351 // Callback should be triggered immediately after StartUpdating is called
352 // when start_count_ is reset to 0.
353 manager.StartUpdating();
354 Thread::Current()->ProcessMessages(0);
355 EXPECT_TRUE(callback_called_);
356}
357
358// Verify that MergeNetworkList() merges network lists properly.
359TEST_F(NetworkTest, TestBasicMergeNetworkList) {
360 Network ipv4_network1("test_eth0", "Test Network Adapter 1",
361 IPAddress(0x12345600U), 24);
362 Network ipv4_network2("test_eth1", "Test Network Adapter 2",
363 IPAddress(0x00010000U), 16);
364 ipv4_network1.AddIP(IPAddress(0x12345678));
365 ipv4_network2.AddIP(IPAddress(0x00010004));
366 BasicNetworkManager manager;
367
368 // Add ipv4_network1 to the list of networks.
369 NetworkManager::NetworkList list;
370 list.push_back(new Network(ipv4_network1));
371 bool changed;
guoweis@webrtc.orga094cac2015-01-28 19:34:05 +0000372 NetworkManager::Stats stats = MergeNetworkList(manager, list, &changed);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000373 EXPECT_TRUE(changed);
guoweis@webrtc.orga094cac2015-01-28 19:34:05 +0000374 EXPECT_EQ(stats.ipv6_network_count, 0);
375 EXPECT_EQ(stats.ipv4_network_count, 1);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000376 list.clear();
377
378 manager.GetNetworks(&list);
379 EXPECT_EQ(1U, list.size());
Qingsi Wang10a0e512018-05-16 13:37:03 -0700380 EXPECT_TRUE(SameNameAndPrefix(ipv4_network1, *list[0]));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000381 Network* net1 = list[0];
honghaiza0c44ea2016-03-23 16:07:48 -0700382 uint16_t net_id1 = net1->id();
383 EXPECT_EQ(1, net_id1);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000384 list.clear();
385
386 // Replace ipv4_network1 with ipv4_network2.
387 list.push_back(new Network(ipv4_network2));
guoweis@webrtc.orga094cac2015-01-28 19:34:05 +0000388 stats = MergeNetworkList(manager, list, &changed);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000389 EXPECT_TRUE(changed);
guoweis@webrtc.orga094cac2015-01-28 19:34:05 +0000390 EXPECT_EQ(stats.ipv6_network_count, 0);
391 EXPECT_EQ(stats.ipv4_network_count, 1);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000392 list.clear();
393
394 manager.GetNetworks(&list);
395 EXPECT_EQ(1U, list.size());
Qingsi Wang10a0e512018-05-16 13:37:03 -0700396 EXPECT_TRUE(SameNameAndPrefix(ipv4_network2, *list[0]));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000397 Network* net2 = list[0];
honghaiza0c44ea2016-03-23 16:07:48 -0700398 uint16_t net_id2 = net2->id();
399 // Network id will increase.
400 EXPECT_LT(net_id1, net_id2);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000401 list.clear();
402
403 // Add Network2 back.
404 list.push_back(new Network(ipv4_network1));
405 list.push_back(new Network(ipv4_network2));
guoweis@webrtc.orga094cac2015-01-28 19:34:05 +0000406 stats = MergeNetworkList(manager, list, &changed);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000407 EXPECT_TRUE(changed);
guoweis@webrtc.orga094cac2015-01-28 19:34:05 +0000408 EXPECT_EQ(stats.ipv6_network_count, 0);
409 EXPECT_EQ(stats.ipv4_network_count, 2);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000410 list.clear();
411
412 // Verify that we get previous instances of Network objects.
413 manager.GetNetworks(&list);
414 EXPECT_EQ(2U, list.size());
415 EXPECT_TRUE((net1 == list[0] && net2 == list[1]) ||
416 (net1 == list[1] && net2 == list[0]));
honghaiza0c44ea2016-03-23 16:07:48 -0700417 EXPECT_TRUE((net_id1 == list[0]->id() && net_id2 == list[1]->id()) ||
418 (net_id1 == list[1]->id() && net_id2 == list[0]->id()));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000419 list.clear();
420
421 // Call MergeNetworkList() again and verify that we don't get update
422 // notification.
423 list.push_back(new Network(ipv4_network2));
424 list.push_back(new Network(ipv4_network1));
guoweis@webrtc.orga094cac2015-01-28 19:34:05 +0000425 stats = MergeNetworkList(manager, list, &changed);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000426 EXPECT_FALSE(changed);
guoweis@webrtc.orga094cac2015-01-28 19:34:05 +0000427 EXPECT_EQ(stats.ipv6_network_count, 0);
428 EXPECT_EQ(stats.ipv4_network_count, 2);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000429 list.clear();
430
431 // Verify that we get previous instances of Network objects.
432 manager.GetNetworks(&list);
433 EXPECT_EQ(2U, list.size());
434 EXPECT_TRUE((net1 == list[0] && net2 == list[1]) ||
435 (net1 == list[1] && net2 == list[0]));
honghaiza0c44ea2016-03-23 16:07:48 -0700436 EXPECT_TRUE((net_id1 == list[0]->id() && net_id2 == list[1]->id()) ||
437 (net_id1 == list[1]->id() && net_id2 == list[0]->id()));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000438 list.clear();
439}
440
441// Sets up some test IPv6 networks and appends them to list.
442// Four networks are added - public and link local, for two interfaces.
443void SetupNetworks(NetworkManager::NetworkList* list) {
444 IPAddress ip;
445 IPAddress prefix;
guoweis@webrtc.orgbbce5ef2015-03-05 04:38:29 +0000446 EXPECT_TRUE(IPFromString("abcd::1234:5678:abcd:ef12", &ip));
447 EXPECT_TRUE(IPFromString("abcd::", &prefix));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000448 // First, fake link-locals.
449 Network ipv6_eth0_linklocalnetwork("test_eth0", "Test NetworkAdapter 1",
450 prefix, 64);
451 ipv6_eth0_linklocalnetwork.AddIP(ip);
guoweis@webrtc.orgbbce5ef2015-03-05 04:38:29 +0000452 EXPECT_TRUE(IPFromString("abcd::5678:abcd:ef12:3456", &ip));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000453 Network ipv6_eth1_linklocalnetwork("test_eth1", "Test NetworkAdapter 2",
454 prefix, 64);
455 ipv6_eth1_linklocalnetwork.AddIP(ip);
456 // Public networks:
457 EXPECT_TRUE(IPFromString("2401:fa00:4:1000:be30:5bff:fee5:c3", &ip));
458 prefix = TruncateIP(ip, 64);
459 Network ipv6_eth0_publicnetwork1_ip1("test_eth0", "Test NetworkAdapter 1",
460 prefix, 64);
461 ipv6_eth0_publicnetwork1_ip1.AddIP(ip);
462 EXPECT_TRUE(IPFromString("2400:4030:1:2c00:be30:abcd:efab:cdef", &ip));
463 prefix = TruncateIP(ip, 64);
464 Network ipv6_eth1_publicnetwork1_ip1("test_eth1", "Test NetworkAdapter 1",
465 prefix, 64);
466 ipv6_eth1_publicnetwork1_ip1.AddIP(ip);
467 list->push_back(new Network(ipv6_eth0_linklocalnetwork));
468 list->push_back(new Network(ipv6_eth1_linklocalnetwork));
469 list->push_back(new Network(ipv6_eth0_publicnetwork1_ip1));
470 list->push_back(new Network(ipv6_eth1_publicnetwork1_ip1));
471}
472
473// Test that the basic network merging case works.
474TEST_F(NetworkTest, TestIPv6MergeNetworkList) {
475 BasicNetworkManager manager;
Yves Gerey665174f2018-06-19 15:03:05 +0200476 manager.SignalNetworksChanged.connect(static_cast<NetworkTest*>(this),
477 &NetworkTest::OnNetworksChanged);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000478 NetworkManager::NetworkList original_list;
479 SetupNetworks(&original_list);
480 bool changed = false;
guoweis@webrtc.orga094cac2015-01-28 19:34:05 +0000481 NetworkManager::Stats stats =
482 MergeNetworkList(manager, original_list, &changed);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000483 EXPECT_TRUE(changed);
guoweis@webrtc.orga094cac2015-01-28 19:34:05 +0000484 EXPECT_EQ(stats.ipv6_network_count, 4);
485 EXPECT_EQ(stats.ipv4_network_count, 0);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000486 NetworkManager::NetworkList list;
487 manager.GetNetworks(&list);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000488 // Verify that the original members are in the merged list.
Steve Anton2acd1632019-03-25 13:48:30 -0700489 EXPECT_THAT(list, UnorderedElementsAreArray(original_list));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000490}
491
492// Tests that when two network lists that describe the same set of networks are
493// merged, that the changed callback is not called, and that the original
494// objects remain in the result list.
495TEST_F(NetworkTest, TestNoChangeMerge) {
496 BasicNetworkManager manager;
Yves Gerey665174f2018-06-19 15:03:05 +0200497 manager.SignalNetworksChanged.connect(static_cast<NetworkTest*>(this),
498 &NetworkTest::OnNetworksChanged);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000499 NetworkManager::NetworkList original_list;
500 SetupNetworks(&original_list);
501 bool changed = false;
502 MergeNetworkList(manager, original_list, &changed);
503 EXPECT_TRUE(changed);
504 // Second list that describes the same networks but with new objects.
505 NetworkManager::NetworkList second_list;
506 SetupNetworks(&second_list);
507 changed = false;
508 MergeNetworkList(manager, second_list, &changed);
509 EXPECT_FALSE(changed);
510 NetworkManager::NetworkList resulting_list;
511 manager.GetNetworks(&resulting_list);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000512 // Verify that the original members are in the merged list.
Steve Anton2acd1632019-03-25 13:48:30 -0700513 EXPECT_THAT(resulting_list, UnorderedElementsAreArray(original_list));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000514 // Doublecheck that the new networks aren't in the list.
Steve Anton2acd1632019-03-25 13:48:30 -0700515 for (const Network* network : second_list) {
516 EXPECT_THAT(resulting_list, Not(Contains(network)));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000517 }
518}
519
520// Test that we can merge a network that is the same as another network but with
521// a different IP. The original network should remain in the list, but have its
522// IP changed.
523TEST_F(NetworkTest, MergeWithChangedIP) {
524 BasicNetworkManager manager;
Yves Gerey665174f2018-06-19 15:03:05 +0200525 manager.SignalNetworksChanged.connect(static_cast<NetworkTest*>(this),
526 &NetworkTest::OnNetworksChanged);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000527 NetworkManager::NetworkList original_list;
528 SetupNetworks(&original_list);
529 // Make a network that we're going to change.
530 IPAddress ip;
531 EXPECT_TRUE(IPFromString("2401:fa01:4:1000:be30:faa:fee:faa", &ip));
532 IPAddress prefix = TruncateIP(ip, 64);
Yves Gerey665174f2018-06-19 15:03:05 +0200533 Network* network_to_change =
534 new Network("test_eth0", "Test Network Adapter 1", prefix, 64);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000535 Network* changed_network = new Network(*network_to_change);
536 network_to_change->AddIP(ip);
537 IPAddress changed_ip;
538 EXPECT_TRUE(IPFromString("2401:fa01:4:1000:be30:f00:f00:f00", &changed_ip));
539 changed_network->AddIP(changed_ip);
540 original_list.push_back(network_to_change);
541 bool changed = false;
542 MergeNetworkList(manager, original_list, &changed);
543 NetworkManager::NetworkList second_list;
544 SetupNetworks(&second_list);
545 second_list.push_back(changed_network);
546 changed = false;
547 MergeNetworkList(manager, second_list, &changed);
548 EXPECT_TRUE(changed);
549 NetworkManager::NetworkList list;
550 manager.GetNetworks(&list);
551 EXPECT_EQ(original_list.size(), list.size());
552 // Make sure the original network is still in the merged list.
Steve Anton2acd1632019-03-25 13:48:30 -0700553 EXPECT_THAT(list, Contains(network_to_change));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000554 EXPECT_EQ(changed_ip, network_to_change->GetIPs().at(0));
555}
556
557// Testing a similar case to above, but checking that a network can be updated
558// with additional IPs (not just a replacement).
559TEST_F(NetworkTest, TestMultipleIPMergeNetworkList) {
560 BasicNetworkManager manager;
Yves Gerey665174f2018-06-19 15:03:05 +0200561 manager.SignalNetworksChanged.connect(static_cast<NetworkTest*>(this),
562 &NetworkTest::OnNetworksChanged);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000563 NetworkManager::NetworkList original_list;
564 SetupNetworks(&original_list);
565 bool changed = false;
566 MergeNetworkList(manager, original_list, &changed);
567 EXPECT_TRUE(changed);
568 IPAddress ip;
569 IPAddress check_ip;
570 IPAddress prefix;
571 // Add a second IP to the public network on eth0 (2401:fa00:4:1000/64).
572 EXPECT_TRUE(IPFromString("2401:fa00:4:1000:be30:5bff:fee5:c6", &ip));
573 prefix = TruncateIP(ip, 64);
574 Network ipv6_eth0_publicnetwork1_ip2("test_eth0", "Test NetworkAdapter 1",
575 prefix, 64);
576 // This is the IP that already existed in the public network on eth0.
577 EXPECT_TRUE(IPFromString("2401:fa00:4:1000:be30:5bff:fee5:c3", &check_ip));
578 ipv6_eth0_publicnetwork1_ip2.AddIP(ip);
579 original_list.push_back(new Network(ipv6_eth0_publicnetwork1_ip2));
580 changed = false;
581 MergeNetworkList(manager, original_list, &changed);
582 EXPECT_TRUE(changed);
583 // There should still be four networks.
584 NetworkManager::NetworkList list;
585 manager.GetNetworks(&list);
586 EXPECT_EQ(4U, list.size());
587 // Check the gathered IPs.
588 int matchcount = 0;
589 for (NetworkManager::NetworkList::iterator it = list.begin();
590 it != list.end(); ++it) {
Qingsi Wang10a0e512018-05-16 13:37:03 -0700591 if (SameNameAndPrefix(**it, *original_list[2])) {
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000592 ++matchcount;
593 EXPECT_EQ(1, matchcount);
594 // This should be the same network object as before.
595 EXPECT_EQ((*it), original_list[2]);
596 // But with two addresses now.
Steve Anton2acd1632019-03-25 13:48:30 -0700597 EXPECT_THAT((*it)->GetIPs(),
598 UnorderedElementsAre(InterfaceAddress(check_ip),
599 InterfaceAddress(ip)));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000600 } else {
601 // Check the IP didn't get added anywhere it wasn't supposed to.
Steve Anton2acd1632019-03-25 13:48:30 -0700602 EXPECT_THAT((*it)->GetIPs(), Not(Contains(InterfaceAddress(ip))));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000603 }
604 }
605}
606
607// Test that merge correctly distinguishes multiple networks on an interface.
608TEST_F(NetworkTest, TestMultiplePublicNetworksOnOneInterfaceMerge) {
609 BasicNetworkManager manager;
Yves Gerey665174f2018-06-19 15:03:05 +0200610 manager.SignalNetworksChanged.connect(static_cast<NetworkTest*>(this),
611 &NetworkTest::OnNetworksChanged);
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000612 NetworkManager::NetworkList original_list;
613 SetupNetworks(&original_list);
614 bool changed = false;
615 MergeNetworkList(manager, original_list, &changed);
616 EXPECT_TRUE(changed);
617 IPAddress ip;
618 IPAddress prefix;
619 // A second network for eth0.
620 EXPECT_TRUE(IPFromString("2400:4030:1:2c00:be30:5bff:fee5:c3", &ip));
621 prefix = TruncateIP(ip, 64);
622 Network ipv6_eth0_publicnetwork2_ip1("test_eth0", "Test NetworkAdapter 1",
623 prefix, 64);
624 ipv6_eth0_publicnetwork2_ip1.AddIP(ip);
625 original_list.push_back(new Network(ipv6_eth0_publicnetwork2_ip1));
626 changed = false;
627 MergeNetworkList(manager, original_list, &changed);
628 EXPECT_TRUE(changed);
629 // There should be five networks now.
630 NetworkManager::NetworkList list;
631 manager.GetNetworks(&list);
632 EXPECT_EQ(5U, list.size());
633 // Check the resulting addresses.
634 for (NetworkManager::NetworkList::iterator it = list.begin();
635 it != list.end(); ++it) {
636 if ((*it)->prefix() == ipv6_eth0_publicnetwork2_ip1.prefix() &&
637 (*it)->name() == ipv6_eth0_publicnetwork2_ip1.name()) {
638 // Check the new network has 1 IP and that it's the correct one.
639 EXPECT_EQ(1U, (*it)->GetIPs().size());
640 EXPECT_EQ(ip, (*it)->GetIPs().at(0));
641 } else {
642 // Check the IP didn't get added anywhere it wasn't supposed to.
Steve Anton2acd1632019-03-25 13:48:30 -0700643 EXPECT_THAT((*it)->GetIPs(), Not(Contains(InterfaceAddress(ip))));
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000644 }
645 }
646}
647
honghaizdb8cf502015-12-21 13:08:46 -0800648// Test that DumpNetworks does not crash.
649TEST_F(NetworkTest, TestCreateAndDumpNetworks) {
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000650 BasicNetworkManager manager;
honghaizdb8cf502015-12-21 13:08:46 -0800651 NetworkManager::NetworkList list = GetNetworks(manager, true);
652 bool changed;
653 MergeNetworkList(manager, list, &changed);
654 manager.DumpNetworks();
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000655}
656
Mirko Bonadei44f0f872019-01-20 18:16:42 +0100657TEST_F(NetworkTest, TestIPv6Toggle) {
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000658 BasicNetworkManager manager;
659 bool ipv6_found = false;
660 NetworkManager::NetworkList list;
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000661 list = GetNetworks(manager, true);
662 for (NetworkManager::NetworkList::iterator it = list.begin();
663 it != list.end(); ++it) {
664 if ((*it)->prefix().family() == AF_INET6) {
665 ipv6_found = true;
666 break;
667 }
668 }
669 EXPECT_TRUE(ipv6_found);
670 for (NetworkManager::NetworkList::iterator it = list.begin();
671 it != list.end(); ++it) {
672 delete (*it);
673 }
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000674}
675
deadbeef3427f532017-07-26 16:09:33 -0700676// Test that when network interfaces are sorted and given preference values,
677// IPv6 comes first.
678TEST_F(NetworkTest, IPv6NetworksPreferredOverIPv4) {
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000679 BasicNetworkManager manager;
680 Network ipv4_network1("test_eth0", "Test Network Adapter 1",
681 IPAddress(0x12345600U), 24);
682 ipv4_network1.AddIP(IPAddress(0x12345600U));
683
684 IPAddress ip;
685 IPAddress prefix;
686 EXPECT_TRUE(IPFromString("2400:4030:1:2c00:be30:abcd:efab:cdef", &ip));
687 prefix = TruncateIP(ip, 64);
688 Network ipv6_eth1_publicnetwork1_ip1("test_eth1", "Test NetworkAdapter 2",
689 prefix, 64);
690 ipv6_eth1_publicnetwork1_ip1.AddIP(ip);
691
692 NetworkManager::NetworkList list;
693 list.push_back(new Network(ipv4_network1));
694 list.push_back(new Network(ipv6_eth1_publicnetwork1_ip1));
695 Network* net1 = list[0];
696 Network* net2 = list[1];
697
698 bool changed = false;
699 MergeNetworkList(manager, list, &changed);
700 ASSERT_TRUE(changed);
701 // After sorting IPv6 network should be higher order than IPv4 networks.
702 EXPECT_TRUE(net1->preference() < net2->preference());
703}
704
deadbeef3427f532017-07-26 16:09:33 -0700705// When two interfaces are equivalent in everything but name, they're expected
706// to be preference-ordered by name. For example, "eth0" before "eth1".
707TEST_F(NetworkTest, NetworksSortedByInterfaceName) {
708 BasicNetworkManager manager;
709 Network* eth0 = new Network("test_eth0", "Test Network Adapter 1",
710 IPAddress(0x65432100U), 24);
711 eth0->AddIP(IPAddress(0x65432100U));
712 Network* eth1 = new Network("test_eth1", "Test Network Adapter 2",
713 IPAddress(0x12345600U), 24);
714 eth1->AddIP(IPAddress(0x12345600U));
715 NetworkManager::NetworkList list;
716 // Add them to the list in the opposite of the expected sorted order, to
717 // ensure sorting actually occurs.
718 list.push_back(eth1);
719 list.push_back(eth0);
720
721 bool changed = false;
722 MergeNetworkList(manager, list, &changed);
723 ASSERT_TRUE(changed);
724 // "test_eth0" should be preferred over "test_eth1".
725 EXPECT_TRUE(eth0->preference() > eth1->preference());
726}
727
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000728TEST_F(NetworkTest, TestNetworkAdapterTypes) {
729 Network wifi("wlan0", "Wireless Adapter", IPAddress(0x12345600U), 24,
730 ADAPTER_TYPE_WIFI);
731 EXPECT_EQ(ADAPTER_TYPE_WIFI, wifi.type());
732 Network ethernet("eth0", "Ethernet", IPAddress(0x12345600U), 24,
733 ADAPTER_TYPE_ETHERNET);
734 EXPECT_EQ(ADAPTER_TYPE_ETHERNET, ethernet.type());
735 Network cellular("test_cell", "Cellular Adapter", IPAddress(0x12345600U), 24,
736 ADAPTER_TYPE_CELLULAR);
737 EXPECT_EQ(ADAPTER_TYPE_CELLULAR, cellular.type());
738 Network vpn("bridge_test", "VPN Adapter", IPAddress(0x12345600U), 24,
739 ADAPTER_TYPE_VPN);
740 EXPECT_EQ(ADAPTER_TYPE_VPN, vpn.type());
741 Network unknown("test", "Test Adapter", IPAddress(0x12345600U), 24,
742 ADAPTER_TYPE_UNKNOWN);
743 EXPECT_EQ(ADAPTER_TYPE_UNKNOWN, unknown.type());
744}
745
746#if defined(WEBRTC_POSIX)
747// Verify that we correctly handle interfaces with no address.
748TEST_F(NetworkTest, TestConvertIfAddrsNoAddress) {
749 ifaddrs list;
750 memset(&list, 0, sizeof(list));
751 list.ifa_name = const_cast<char*>("test_iface");
752
753 NetworkManager::NetworkList result;
754 BasicNetworkManager manager;
755 CallConvertIfAddrs(manager, &list, true, &result);
756 EXPECT_TRUE(result.empty());
757}
honghaizdb8cf502015-12-21 13:08:46 -0800758
759// Verify that if there are two addresses on one interface, only one network
760// is generated.
761TEST_F(NetworkTest, TestConvertIfAddrsMultiAddressesOnOneInterface) {
762 char if_name[20] = "rmnet0";
763 ifaddrs* list = nullptr;
764 list = AddIpv6Address(list, if_name, "1000:2000:3000:4000:0:0:0:1",
765 "FFFF:FFFF:FFFF:FFFF::", 0);
766 list = AddIpv6Address(list, if_name, "1000:2000:3000:4000:0:0:0:2",
767 "FFFF:FFFF:FFFF:FFFF::", 0);
768 NetworkManager::NetworkList result;
769 BasicNetworkManager manager;
770 CallConvertIfAddrs(manager, list, true, &result);
771 EXPECT_EQ(1U, result.size());
772 bool changed;
773 // This ensures we release the objects created in CallConvertIfAddrs.
774 MergeNetworkList(manager, result, &changed);
775 ReleaseIfAddrs(list);
776}
Guo-wei Shieh9faf1542015-12-28 14:06:55 -0800777
778TEST_F(NetworkTest, TestConvertIfAddrsNotRunning) {
779 ifaddrs list;
780 memset(&list, 0, sizeof(list));
781 list.ifa_name = const_cast<char*>("test_iface");
782 sockaddr ifa_addr;
783 sockaddr ifa_netmask;
784 list.ifa_addr = &ifa_addr;
785 list.ifa_netmask = &ifa_netmask;
786
787 NetworkManager::NetworkList result;
788 BasicNetworkManager manager;
789 CallConvertIfAddrs(manager, &list, true, &result);
790 EXPECT_TRUE(result.empty());
791}
Honghai Zhang351d77b2016-05-20 15:08:29 -0700792
793// Tests that the network type can be updated after the network monitor is
794// started.
795TEST_F(NetworkTest, TestGetAdapterTypeFromNetworkMonitor) {
796 char if_name1[20] = "wifi0";
797 std::string ipv6_address1 = "1000:2000:3000:4000:0:0:0:1";
798 std::string ipv6_address2 = "1000:2000:3000:8000:0:0:0:1";
799 std::string ipv6_mask = "FFFF:FFFF:FFFF:FFFF::";
800 BasicNetworkManager manager;
801 // A network created before the network monitor is started will get
802 // UNKNOWN type.
803 ifaddrs* addr_list =
804 InstallIpv6Network(if_name1, ipv6_address1, ipv6_mask, manager);
805 EXPECT_EQ(ADAPTER_TYPE_UNKNOWN, GetAdapterType(manager));
806 ReleaseIfAddrs(addr_list);
807 // Note: Do not call ClearNetworks here in order to test that the type
808 // of an existing network can be changed after the network monitor starts
809 // and detects the network type correctly.
810
811 // After the network monitor starts, the type will be updated.
812 FakeNetworkMonitorFactory* factory = new FakeNetworkMonitorFactory();
813 NetworkMonitorFactory::SetFactory(factory);
814 // This brings up the hook with the network monitor.
815 manager.StartUpdating();
816 // Add the same ipv6 address as before but it has the right network type
817 // detected by the network monitor now.
818 addr_list = InstallIpv6Network(if_name1, ipv6_address1, ipv6_mask, manager);
819 EXPECT_EQ(ADAPTER_TYPE_WIFI, GetAdapterType(manager));
820 ReleaseIfAddrs(addr_list);
821 ClearNetworks(manager);
822
823 // Add another network with the type inferred from the network monitor.
824 char if_name2[20] = "cellular0";
825 addr_list = InstallIpv6Network(if_name2, ipv6_address2, ipv6_mask, manager);
826 EXPECT_EQ(ADAPTER_TYPE_CELLULAR, GetAdapterType(manager));
827 ReleaseIfAddrs(addr_list);
828 ClearNetworks(manager);
829}
830
831// Test that the network type can be determined based on name matching in
832// a few cases. Note that UNKNOWN type for non-matching strings has been tested
833// in the above test.
834TEST_F(NetworkTest, TestGetAdapterTypeFromNameMatching) {
Jeroen de Borst8f096d02019-02-21 13:34:45 -0800835 std::string ipv4_address1 = "192.0.0.121";
836 std::string ipv4_mask = "255.255.255.0";
Honghai Zhang351d77b2016-05-20 15:08:29 -0700837 std::string ipv6_address1 = "1000:2000:3000:4000:0:0:0:1";
838 std::string ipv6_address2 = "1000:2000:3000:8000:0:0:0:1";
839 std::string ipv6_mask = "FFFF:FFFF:FFFF:FFFF::";
840 BasicNetworkManager manager;
841
deadbeef4cd599f2017-07-27 15:05:29 -0700842 // IPSec interface; name is in form "ipsec<index>".
843 char if_name[20] = "ipsec11";
Honghai Zhang351d77b2016-05-20 15:08:29 -0700844 ifaddrs* addr_list =
845 InstallIpv6Network(if_name, ipv6_address1, ipv6_mask, manager);
deadbeef4cd599f2017-07-27 15:05:29 -0700846 EXPECT_EQ(ADAPTER_TYPE_VPN, GetAdapterType(manager));
847 ClearNetworks(manager);
848 ReleaseIfAddrs(addr_list);
Honghai Zhang351d77b2016-05-20 15:08:29 -0700849
Qingsi Wange53ac042018-05-08 11:55:07 -0700850 strcpy(if_name, "lo0");
851 addr_list = InstallIpv6Network(if_name, ipv6_address1, ipv6_mask, manager);
852 EXPECT_EQ(ADAPTER_TYPE_LOOPBACK, GetAdapterType(manager));
853 ClearNetworks(manager);
854 ReleaseIfAddrs(addr_list);
855
856 strcpy(if_name, "eth0");
Jeroen de Borst8f096d02019-02-21 13:34:45 -0800857 addr_list = InstallIpv4Network(if_name, ipv4_address1, ipv4_mask, manager);
Qingsi Wange53ac042018-05-08 11:55:07 -0700858 EXPECT_EQ(ADAPTER_TYPE_ETHERNET, GetAdapterType(manager));
859 ClearNetworks(manager);
860 ReleaseIfAddrs(addr_list);
861
Qingsi Wangc5bc9d62019-09-25 15:03:19 -0700862 strcpy(if_name, "wlan0");
863 addr_list = InstallIpv6Network(if_name, ipv6_address1, ipv6_mask, manager);
864 EXPECT_EQ(ADAPTER_TYPE_WIFI, GetAdapterType(manager));
865 ClearNetworks(manager);
866 ReleaseIfAddrs(addr_list);
867
deadbeef4cd599f2017-07-27 15:05:29 -0700868#if defined(WEBRTC_IOS)
869 strcpy(if_name, "pdp_ip0");
870 addr_list = InstallIpv6Network(if_name, ipv6_address1, ipv6_mask, manager);
Honghai Zhang351d77b2016-05-20 15:08:29 -0700871 EXPECT_EQ(ADAPTER_TYPE_CELLULAR, GetAdapterType(manager));
872 ClearNetworks(manager);
873 ReleaseIfAddrs(addr_list);
874
Honghai Zhang63ab8102016-05-26 20:30:15 -0700875 strcpy(if_name, "en0");
876 addr_list = InstallIpv6Network(if_name, ipv6_address1, ipv6_mask, manager);
877 EXPECT_EQ(ADAPTER_TYPE_WIFI, GetAdapterType(manager));
878 ClearNetworks(manager);
879 ReleaseIfAddrs(addr_list);
880
Honghai Zhang351d77b2016-05-20 15:08:29 -0700881#elif defined(WEBRTC_ANDROID)
deadbeef4cd599f2017-07-27 15:05:29 -0700882 strcpy(if_name, "rmnet0");
883 addr_list = InstallIpv6Network(if_name, ipv6_address1, ipv6_mask, manager);
Honghai Zhang351d77b2016-05-20 15:08:29 -0700884 EXPECT_EQ(ADAPTER_TYPE_CELLULAR, GetAdapterType(manager));
885 ClearNetworks(manager);
886 ReleaseIfAddrs(addr_list);
887
Honghai Zhang351d77b2016-05-20 15:08:29 -0700888 strcpy(if_name, "v4-rmnet_data0");
889 addr_list = InstallIpv6Network(if_name, ipv6_address2, ipv6_mask, manager);
890 EXPECT_EQ(ADAPTER_TYPE_CELLULAR, GetAdapterType(manager));
891 ClearNetworks(manager);
892 ReleaseIfAddrs(addr_list);
Jeroen de Borst8f096d02019-02-21 13:34:45 -0800893
894 strcpy(if_name, "clat4");
895 addr_list = InstallIpv4Network(if_name, ipv4_address1, ipv4_mask, manager);
896 EXPECT_EQ(ADAPTER_TYPE_CELLULAR, GetAdapterType(manager));
897 ClearNetworks(manager);
898 ReleaseIfAddrs(addr_list);
Honghai Zhang351d77b2016-05-20 15:08:29 -0700899#endif
900}
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000901#endif // defined(WEBRTC_POSIX)
902
903#if defined(WEBRTC_LINUX) && !defined(WEBRTC_ANDROID)
904// If you want to test non-default routes, you can do the following on a linux
905// machine:
906// 1) Load the dummy network driver:
907// sudo modprobe dummy
908// sudo ifconfig dummy0 127.0.0.1
909// 2) Run this test and confirm the output says it found a dummy route (and
910// passes).
911// 3) When done:
912// sudo rmmmod dummy
913TEST_F(NetworkTest, TestIgnoreNonDefaultRoutes) {
914 BasicNetworkManager manager;
915 NetworkManager::NetworkList list;
916 list = GetNetworks(manager, false);
917 bool found_dummy = false;
Mirko Bonadei675513b2017-11-09 11:09:25 +0100918 RTC_LOG(LS_INFO) << "Looking for dummy network: ";
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000919 for (NetworkManager::NetworkList::iterator it = list.begin();
920 it != list.end(); ++it) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100921 RTC_LOG(LS_INFO) << " Network name: " << (*it)->name();
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000922 found_dummy |= (*it)->name().find("dummy0") != std::string::npos;
923 }
924 for (NetworkManager::NetworkList::iterator it = list.begin();
925 it != list.end(); ++it) {
926 delete (*it);
927 }
928 if (!found_dummy) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100929 RTC_LOG(LS_INFO) << "No dummy found, quitting.";
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000930 return;
931 }
Mirko Bonadei675513b2017-11-09 11:09:25 +0100932 RTC_LOG(LS_INFO) << "Found dummy, running again while ignoring non-default "
933 << "routes.";
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000934 manager.set_ignore_non_default_routes(true);
935 list = GetNetworks(manager, false);
936 for (NetworkManager::NetworkList::iterator it = list.begin();
937 it != list.end(); ++it) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100938 RTC_LOG(LS_INFO) << " Network name: " << (*it)->name();
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000939 EXPECT_TRUE((*it)->name().find("dummy0") == std::string::npos);
940 }
941 for (NetworkManager::NetworkList::iterator it = list.begin();
942 it != list.end(); ++it) {
943 delete (*it);
944 }
945}
946#endif
947
guoweis@webrtc.org4bbd3c82014-09-09 13:54:45 +0000948// Test MergeNetworkList successfully combines all IPs for the same
949// prefix/length into a single Network.
950TEST_F(NetworkTest, TestMergeNetworkList) {
951 BasicNetworkManager manager;
952 NetworkManager::NetworkList list;
953
954 // Create 2 IPAddress classes with only last digit different.
955 IPAddress ip1, ip2;
956 EXPECT_TRUE(IPFromString("2400:4030:1:2c00:be30:0:0:1", &ip1));
957 EXPECT_TRUE(IPFromString("2400:4030:1:2c00:be30:0:0:2", &ip2));
958
959 // Create 2 networks with the same prefix and length.
960 Network* net1 = new Network("em1", "em1", TruncateIP(ip1, 64), 64);
961 Network* net2 = new Network("em1", "em1", TruncateIP(ip1, 64), 64);
962
963 // Add different IP into each.
964 net1->AddIP(ip1);
965 net2->AddIP(ip2);
966
967 list.push_back(net1);
968 list.push_back(net2);
969 bool changed;
970 MergeNetworkList(manager, list, &changed);
971 EXPECT_TRUE(changed);
972
973 NetworkManager::NetworkList list2;
974 manager.GetNetworks(&list2);
975
976 // Make sure the resulted networklist has only 1 element and 2
977 // IPAddresses.
978 EXPECT_EQ(list2.size(), 1uL);
979 EXPECT_EQ(list2[0]->GetIPs().size(), 2uL);
Taylor Brandstetter01cb5f22018-03-07 15:49:32 -0800980 EXPECT_EQ(list2[0]->GetIPs()[0], InterfaceAddress(ip1));
981 EXPECT_EQ(list2[0]->GetIPs()[1], InterfaceAddress(ip2));
guoweis@webrtc.org4bbd3c82014-09-09 13:54:45 +0000982}
983
honghaizdb8cf502015-12-21 13:08:46 -0800984// Test that MergeNetworkList successfully detects the change if
985// a network becomes inactive and then active again.
986TEST_F(NetworkTest, TestMergeNetworkListWithInactiveNetworks) {
987 BasicNetworkManager manager;
988 Network network1("test_wifi", "Test Network Adapter 1",
989 IPAddress(0x12345600U), 24);
990 Network network2("test_eth0", "Test Network Adapter 2",
991 IPAddress(0x00010000U), 16);
992 network1.AddIP(IPAddress(0x12345678));
993 network2.AddIP(IPAddress(0x00010004));
994 NetworkManager::NetworkList list;
995 Network* net1 = new Network(network1);
996 list.push_back(net1);
997 bool changed;
998 MergeNetworkList(manager, list, &changed);
999 EXPECT_TRUE(changed);
1000 list.clear();
1001 manager.GetNetworks(&list);
1002 ASSERT_EQ(1U, list.size());
1003 EXPECT_EQ(net1, list[0]);
1004
1005 list.clear();
1006 Network* net2 = new Network(network2);
1007 list.push_back(net2);
1008 MergeNetworkList(manager, list, &changed);
1009 EXPECT_TRUE(changed);
1010 list.clear();
1011 manager.GetNetworks(&list);
1012 ASSERT_EQ(1U, list.size());
1013 EXPECT_EQ(net2, list[0]);
1014
1015 // Now network1 is inactive. Try to merge it again.
1016 list.clear();
1017 list.push_back(new Network(network1));
1018 MergeNetworkList(manager, list, &changed);
1019 EXPECT_TRUE(changed);
1020 list.clear();
1021 manager.GetNetworks(&list);
1022 ASSERT_EQ(1U, list.size());
1023 EXPECT_TRUE(list[0]->active());
1024 EXPECT_EQ(net1, list[0]);
1025}
1026
guoweis@webrtc.org369a6372014-09-17 22:37:29 +00001027// Test that the filtering logic follows the defined ruleset in network.h.
1028TEST_F(NetworkTest, TestIPv6Selection) {
1029 InterfaceAddress ip;
1030 std::string ipstr;
1031
1032 ipstr = "2401:fa00:4:1000:be30:5bff:fee5:c3";
1033 ASSERT_TRUE(IPFromString(ipstr, IPV6_ADDRESS_FLAG_DEPRECATED, &ip));
1034
1035 // Create a network with this prefix.
Yves Gerey665174f2018-06-19 15:03:05 +02001036 Network ipv6_network("test_eth0", "Test NetworkAdapter", TruncateIP(ip, 64),
1037 64);
guoweis@webrtc.org369a6372014-09-17 22:37:29 +00001038
1039 // When there is no address added, it should return an unspecified
1040 // address.
1041 EXPECT_EQ(ipv6_network.GetBestIP(), IPAddress());
1042 EXPECT_TRUE(IPIsUnspec(ipv6_network.GetBestIP()));
1043
1044 // Deprecated one should not be returned.
1045 ipv6_network.AddIP(ip);
1046 EXPECT_EQ(ipv6_network.GetBestIP(), IPAddress());
1047
aluebs@webrtc.org07dcf602015-02-27 18:42:22 +00001048 // Add ULA one. ULA is unique local address which is starting either
1049 // with 0xfc or 0xfd.
guoweis@webrtc.org369a6372014-09-17 22:37:29 +00001050 ipstr = "fd00:fa00:4:1000:be30:5bff:fee5:c4";
1051 ASSERT_TRUE(IPFromString(ipstr, IPV6_ADDRESS_FLAG_NONE, &ip));
1052 ipv6_network.AddIP(ip);
aluebs@webrtc.org07dcf602015-02-27 18:42:22 +00001053 EXPECT_EQ(ipv6_network.GetBestIP(), static_cast<IPAddress>(ip));
guoweis@webrtc.org369a6372014-09-17 22:37:29 +00001054
aluebs@webrtc.org07dcf602015-02-27 18:42:22 +00001055 // Add global one.
guoweis@webrtc.org369a6372014-09-17 22:37:29 +00001056 ipstr = "2401:fa00:4:1000:be30:5bff:fee5:c5";
1057 ASSERT_TRUE(IPFromString(ipstr, IPV6_ADDRESS_FLAG_NONE, &ip));
1058 ipv6_network.AddIP(ip);
aluebs@webrtc.org07dcf602015-02-27 18:42:22 +00001059 EXPECT_EQ(ipv6_network.GetBestIP(), static_cast<IPAddress>(ip));
guoweis@webrtc.org369a6372014-09-17 22:37:29 +00001060
1061 // Add global dynamic temporary one.
1062 ipstr = "2401:fa00:4:1000:be30:5bff:fee5:c6";
1063 ASSERT_TRUE(IPFromString(ipstr, IPV6_ADDRESS_FLAG_TEMPORARY, &ip));
1064 ipv6_network.AddIP(ip);
1065 EXPECT_EQ(ipv6_network.GetBestIP(), static_cast<IPAddress>(ip));
1066}
1067
honghaiz023f3ef2015-10-19 09:39:32 -07001068TEST_F(NetworkTest, TestNetworkMonitoring) {
1069 BasicNetworkManager manager;
1070 manager.SignalNetworksChanged.connect(static_cast<NetworkTest*>(this),
1071 &NetworkTest::OnNetworksChanged);
1072 FakeNetworkMonitorFactory* factory = new FakeNetworkMonitorFactory();
1073 NetworkMonitorFactory::SetFactory(factory);
1074 manager.StartUpdating();
honghaizcec0a082016-01-15 14:49:09 -08001075 FakeNetworkMonitor* network_monitor = GetNetworkMonitor(manager);
1076 EXPECT_TRUE(network_monitor && network_monitor->started());
honghaiz023f3ef2015-10-19 09:39:32 -07001077 EXPECT_TRUE_WAIT(callback_called_, 1000);
1078 callback_called_ = false;
1079
1080 // Clear the networks so that there will be network changes below.
1081 ClearNetworks(manager);
1082 // Network manager is started, so the callback is called when the network
1083 // monitor fires the network-change event.
1084 network_monitor->OnNetworksChanged();
1085 EXPECT_TRUE_WAIT(callback_called_, 1000);
1086
honghaizcec0a082016-01-15 14:49:09 -08001087 // Network manager is stopped.
honghaiz023f3ef2015-10-19 09:39:32 -07001088 manager.StopUpdating();
honghaizcec0a082016-01-15 14:49:09 -08001089 EXPECT_FALSE(GetNetworkMonitor(manager)->started());
honghaiz023f3ef2015-10-19 09:39:32 -07001090
1091 NetworkMonitorFactory::ReleaseFactory(factory);
1092}
1093
Edward Lemur8dc945c2016-07-21 10:16:40 +02001094// Fails on Android: https://bugs.chromium.org/p/webrtc/issues/detail?id=4364.
1095#if defined(WEBRTC_ANDROID)
1096#define MAYBE_DefaultLocalAddress DISABLED_DefaultLocalAddress
1097#else
1098#define MAYBE_DefaultLocalAddress DefaultLocalAddress
1099#endif
1100TEST_F(NetworkTest, MAYBE_DefaultLocalAddress) {
Guo-wei Shieha34c39e2015-11-25 13:12:26 -08001101 IPAddress ip;
guoweis56271ed2016-01-15 14:45:06 -08001102 TestBasicNetworkManager manager;
1103 manager.SignalNetworksChanged.connect(static_cast<NetworkTest*>(this),
1104 &NetworkTest::OnNetworksChanged);
1105 FakeNetworkMonitorFactory* factory = new FakeNetworkMonitorFactory();
1106 NetworkMonitorFactory::SetFactory(factory);
1107 manager.StartUpdating();
1108 EXPECT_TRUE_WAIT(callback_called_, 1000);
Guo-wei Shieha34c39e2015-11-25 13:12:26 -08001109
1110 // Make sure we can query default local address when an address for such
1111 // address family exists.
Guo-wei Shieh9af97f82015-11-10 14:47:39 -08001112 std::vector<Network*> networks;
1113 manager.GetNetworks(&networks);
guoweis56271ed2016-01-15 14:45:06 -08001114 EXPECT_TRUE(!networks.empty());
Steve Anton9de3aac2017-10-24 10:08:26 -07001115 for (const auto* network : networks) {
Guo-wei Shieh9af97f82015-11-10 14:47:39 -08001116 if (network->GetBestIP().family() == AF_INET) {
1117 EXPECT_TRUE(manager.QueryDefaultLocalAddress(AF_INET) != IPAddress());
guoweis56271ed2016-01-15 14:45:06 -08001118 } else if (network->GetBestIP().family() == AF_INET6 &&
1119 !IPIsLoopback(network->GetBestIP())) {
1120 // Existence of an IPv6 loopback address doesn't mean it has IPv6 network
1121 // enabled.
Guo-wei Shieh9af97f82015-11-10 14:47:39 -08001122 EXPECT_TRUE(manager.QueryDefaultLocalAddress(AF_INET6) != IPAddress());
1123 }
1124 }
Guo-wei Shieha34c39e2015-11-25 13:12:26 -08001125
1126 // GetDefaultLocalAddress should return the valid default address after set.
1127 manager.set_default_local_addresses(GetLoopbackIP(AF_INET),
1128 GetLoopbackIP(AF_INET6));
1129 EXPECT_TRUE(manager.GetDefaultLocalAddress(AF_INET, &ip));
1130 EXPECT_EQ(ip, GetLoopbackIP(AF_INET));
1131 EXPECT_TRUE(manager.GetDefaultLocalAddress(AF_INET6, &ip));
1132 EXPECT_EQ(ip, GetLoopbackIP(AF_INET6));
honghaizaf83fe62016-04-18 14:50:44 -07001133
1134 // More tests on GetDefaultLocalAddress with ipv6 addresses where the set
1135 // default address may be different from the best IP address of any network.
1136 InterfaceAddress ip1;
1137 EXPECT_TRUE(IPFromString("abcd::1234:5678:abcd:1111",
1138 IPV6_ADDRESS_FLAG_TEMPORARY, &ip1));
1139 // Create a network with a prefix of ip1.
1140 Network ipv6_network("test_eth0", "Test NetworkAdapter", TruncateIP(ip1, 64),
1141 64);
1142 IPAddress ip2;
1143 EXPECT_TRUE(IPFromString("abcd::1234:5678:abcd:2222", &ip2));
1144 ipv6_network.AddIP(ip1);
1145 ipv6_network.AddIP(ip2);
1146 BasicNetworkManager::NetworkList list(1, new Network(ipv6_network));
1147 bool changed;
1148 MergeNetworkList(manager, list, &changed);
1149 // If the set default address is not in any network, GetDefaultLocalAddress
1150 // should return it.
1151 IPAddress ip3;
1152 EXPECT_TRUE(IPFromString("abcd::1234:5678:abcd:3333", &ip3));
1153 manager.set_default_local_addresses(GetLoopbackIP(AF_INET), ip3);
1154 EXPECT_TRUE(manager.GetDefaultLocalAddress(AF_INET6, &ip));
1155 EXPECT_EQ(ip3, ip);
1156 // If the set default address is in a network, GetDefaultLocalAddress will
1157 // return the best IP in that network.
1158 manager.set_default_local_addresses(GetLoopbackIP(AF_INET), ip2);
1159 EXPECT_TRUE(manager.GetDefaultLocalAddress(AF_INET6, &ip));
1160 EXPECT_EQ(static_cast<IPAddress>(ip1), ip);
1161
Guo-wei Shieh9af97f82015-11-10 14:47:39 -08001162 manager.StopUpdating();
1163}
1164
henrike@webrtc.orgf0488722014-05-13 18:00:26 +00001165} // namespace rtc