OLD | NEW |
1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "net/base/network_quality_estimator.h" | 5 #include "net/base/network_quality_estimator.h" |
6 | 6 |
7 #include <stdint.h> | 7 #include <stdint.h> |
8 | 8 |
9 #include <limits> | 9 #include <limits> |
10 #include <map> | 10 #include <map> |
11 | 11 |
12 #include "base/basictypes.h" | 12 #include "base/basictypes.h" |
13 #include "base/files/file_path.h" | |
14 #include "base/logging.h" | 13 #include "base/logging.h" |
15 #include "base/memory/scoped_ptr.h" | 14 #include "base/memory/scoped_ptr.h" |
16 #include "base/metrics/histogram_samples.h" | 15 #include "base/metrics/histogram_samples.h" |
17 #include "base/run_loop.h" | 16 #include "base/run_loop.h" |
18 #include "base/strings/string_number_conversions.h" | 17 #include "base/strings/string_number_conversions.h" |
19 #include "base/test/histogram_tester.h" | 18 #include "base/test/histogram_tester.h" |
20 #include "base/time/time.h" | 19 #include "base/time/time.h" |
21 #include "build/build_config.h" | 20 #include "build/build_config.h" |
22 #include "net/base/external_estimate_provider.h" | 21 #include "net/base/external_estimate_provider.h" |
23 #include "net/base/load_flags.h" | 22 #include "net/base/load_flags.h" |
24 #include "net/base/network_change_notifier.h" | 23 #include "net/base/network_change_notifier.h" |
25 #include "net/http/http_status_code.h" | 24 #include "net/test/embedded_test_server/embedded_test_server.h" |
26 #include "net/test/spawned_test_server/spawned_test_server.h" | |
27 #include "net/url_request/url_request_test_util.h" | 25 #include "net/url_request/url_request_test_util.h" |
28 #include "testing/gtest/include/gtest/gtest.h" | 26 #include "testing/gtest/include/gtest/gtest.h" |
29 #include "url/gurl.h" | 27 #include "url/gurl.h" |
30 | 28 |
31 namespace { | 29 namespace { |
32 | 30 |
33 #if !defined(OS_IOS) | |
34 // SpawnedTestServer is not supported on iOS. | |
35 // Less verbose way of running a simple testserver for the tests below. | |
36 class LocalHttpTestServer : public net::SpawnedTestServer { | |
37 public: | |
38 LocalHttpTestServer() | |
39 : net::SpawnedTestServer(net::SpawnedTestServer::TYPE_HTTP, | |
40 net::SpawnedTestServer::kLocalhost, | |
41 base::FilePath()) {} | |
42 }; | |
43 #endif // !defined(OS_IOS) | |
44 | |
45 // Helps in setting the current network type and id. | 31 // Helps in setting the current network type and id. |
46 class TestNetworkQualityEstimator : public net::NetworkQualityEstimator { | 32 class TestNetworkQualityEstimator : public net::NetworkQualityEstimator { |
47 public: | 33 public: |
48 TestNetworkQualityEstimator( | 34 TestNetworkQualityEstimator( |
49 const std::map<std::string, std::string>& variation_params, | 35 const std::map<std::string, std::string>& variation_params) |
50 scoped_ptr<net::ExternalEstimateProvider> external_estimate_provider) | 36 : NetworkQualityEstimator(scoped_ptr<net::ExternalEstimateProvider>(), |
51 : NetworkQualityEstimator(external_estimate_provider.Pass(), | |
52 variation_params, | 37 variation_params, |
53 true, | 38 true, |
54 true) { | 39 true) {} |
55 #if !defined(OS_IOS) | |
56 // Set up test server. | |
57 DCHECK(test_server_.Start()); | |
58 #endif // !defined(OS_IOS) | |
59 } | |
60 | |
61 explicit TestNetworkQualityEstimator( | |
62 const std::map<std::string, std::string>& variation_params) | |
63 : TestNetworkQualityEstimator( | |
64 variation_params, | |
65 scoped_ptr<net::ExternalEstimateProvider>()) {} | |
66 | 40 |
67 ~TestNetworkQualityEstimator() override {} | 41 ~TestNetworkQualityEstimator() override {} |
68 | 42 |
69 // Overrides the current network type and id. | 43 // Overrides the current network type and id. |
70 // Notifies network quality estimator of change in connection. | 44 // Notifies network quality estimator of change in connection. |
71 void SimulateNetworkChangeTo(net::NetworkChangeNotifier::ConnectionType type, | 45 void SimulateNetworkChangeTo(net::NetworkChangeNotifier::ConnectionType type, |
72 std::string network_id) { | 46 std::string network_id) { |
73 current_network_type_ = type; | 47 current_network_type_ = type; |
74 current_network_id_ = network_id; | 48 current_network_id_ = network_id; |
75 OnConnectionTypeChanged(type); | 49 OnConnectionTypeChanged(type); |
76 } | 50 } |
77 | 51 |
78 #if !defined(OS_IOS) | |
79 // Returns a GURL hosted at embedded test server. | |
80 const GURL GetEchoURL() const { return test_server_.GetURL("/echo.html"); } | |
81 #endif // !defined(OS_IOS) | |
82 | |
83 using NetworkQualityEstimator::ReadCachedNetworkQualityEstimate; | 52 using NetworkQualityEstimator::ReadCachedNetworkQualityEstimate; |
84 using NetworkQualityEstimator::OnConnectionTypeChanged; | 53 using NetworkQualityEstimator::OnConnectionTypeChanged; |
85 | 54 |
86 private: | 55 private: |
87 // NetworkQualityEstimator implementation that returns the overridden network | 56 // NetworkQualityEstimator implementation that returns the overridden network |
88 // id (instead of invoking platform APIs). | 57 // id (instead of invoking platform APIs). |
89 NetworkQualityEstimator::NetworkID GetCurrentNetworkID() const override { | 58 NetworkQualityEstimator::NetworkID GetCurrentNetworkID() const override { |
90 return NetworkQualityEstimator::NetworkID(current_network_type_, | 59 return NetworkQualityEstimator::NetworkID(current_network_type_, |
91 current_network_id_); | 60 current_network_id_); |
92 } | 61 } |
93 | 62 |
94 net::NetworkChangeNotifier::ConnectionType current_network_type_; | 63 net::NetworkChangeNotifier::ConnectionType current_network_type_; |
95 std::string current_network_id_; | 64 std::string current_network_id_; |
96 | |
97 #if !defined(OS_IOS) | |
98 // Test server used for testing. | |
99 LocalHttpTestServer test_server_; | |
100 #endif // !defined(OS_IOS) | |
101 | |
102 DISALLOW_COPY_AND_ASSIGN(TestNetworkQualityEstimator); | |
103 }; | 65 }; |
104 | 66 |
105 } // namespace | 67 } // namespace |
106 | 68 |
107 namespace net { | 69 namespace net { |
108 | 70 |
109 #if !defined(OS_IOS) | |
110 // SpawnedTestServer is not supported on iOS. | |
111 TEST(NetworkQualityEstimatorTest, TestKbpsRTTUpdates) { | 71 TEST(NetworkQualityEstimatorTest, TestKbpsRTTUpdates) { |
| 72 net::test_server::EmbeddedTestServer embedded_test_server; |
| 73 embedded_test_server.ServeFilesFromDirectory( |
| 74 base::FilePath(FILE_PATH_LITERAL("net/data/url_request_unittest"))); |
| 75 ASSERT_TRUE(embedded_test_server.InitializeAndWaitUntilReady()); |
| 76 |
112 base::HistogramTester histogram_tester; | 77 base::HistogramTester histogram_tester; |
113 // Enable requests to local host to be used for network quality estimation. | 78 // Enable requests to local host to be used for network quality estimation. |
114 std::map<std::string, std::string> variation_params; | 79 std::map<std::string, std::string> variation_params; |
115 TestNetworkQualityEstimator estimator(variation_params); | 80 TestNetworkQualityEstimator estimator(variation_params); |
116 | 81 |
117 EXPECT_EQ(NetworkQualityEstimator::InvalidRTT(), | 82 EXPECT_EQ(NetworkQualityEstimator::InvalidRTT(), |
118 estimator.GetRTTEstimateInternal(base::TimeTicks(), 100)); | 83 estimator.GetRTTEstimateInternal(base::TimeTicks(), 100)); |
119 EXPECT_EQ(NetworkQualityEstimator::kInvalidThroughput, | 84 EXPECT_EQ(NetworkQualityEstimator::kInvalidThroughput, |
120 estimator.GetDownlinkThroughputKbpsEstimateInternal( | 85 estimator.GetDownlinkThroughputKbpsEstimateInternal( |
121 base::TimeTicks(), 100)); | 86 base::TimeTicks(), 100)); |
122 | 87 |
123 TestDelegate test_delegate; | 88 TestDelegate test_delegate; |
124 TestURLRequestContext context(true); | 89 TestURLRequestContext context(false); |
125 context.set_network_quality_estimator(&estimator); | |
126 context.Init(); | |
127 | 90 |
128 scoped_ptr<URLRequest> request(context.CreateRequest( | 91 scoped_ptr<URLRequest> request( |
129 estimator.GetEchoURL(), DEFAULT_PRIORITY, &test_delegate)); | 92 context.CreateRequest(embedded_test_server.GetURL("/echo.html"), |
| 93 DEFAULT_PRIORITY, &test_delegate)); |
130 request->SetLoadFlags(request->load_flags() | LOAD_MAIN_FRAME); | 94 request->SetLoadFlags(request->load_flags() | LOAD_MAIN_FRAME); |
131 request->Start(); | 95 request->Start(); |
| 96 |
132 base::RunLoop().Run(); | 97 base::RunLoop().Run(); |
133 | 98 |
134 // Both RTT and downstream throughput should be updated. | 99 // Both RTT and downstream throughput should be updated. |
| 100 estimator.NotifyHeadersReceived(*request); |
| 101 estimator.NotifyRequestCompleted(*request); |
135 EXPECT_NE(NetworkQualityEstimator::InvalidRTT(), | 102 EXPECT_NE(NetworkQualityEstimator::InvalidRTT(), |
136 estimator.GetRTTEstimateInternal(base::TimeTicks(), 100)); | 103 estimator.GetRTTEstimateInternal(base::TimeTicks(), 100)); |
137 EXPECT_NE(NetworkQualityEstimator::kInvalidThroughput, | 104 EXPECT_NE(NetworkQualityEstimator::kInvalidThroughput, |
138 estimator.GetDownlinkThroughputKbpsEstimateInternal( | 105 estimator.GetDownlinkThroughputKbpsEstimateInternal( |
139 base::TimeTicks(), 100)); | 106 base::TimeTicks(), 100)); |
140 | 107 |
141 base::TimeDelta rtt = NetworkQualityEstimator::InvalidRTT(); | 108 base::TimeDelta rtt = NetworkQualityEstimator::InvalidRTT(); |
142 int32_t kbps = NetworkQualityEstimator::kInvalidThroughput; | 109 int32_t kbps = NetworkQualityEstimator::kInvalidThroughput; |
143 EXPECT_TRUE(estimator.GetRTTEstimate(&rtt)); | 110 EXPECT_TRUE(estimator.GetRTTEstimate(&rtt)); |
144 EXPECT_TRUE(estimator.GetDownlinkThroughputKbpsEstimate(&kbps)); | 111 EXPECT_TRUE(estimator.GetDownlinkThroughputKbpsEstimate(&kbps)); |
145 EXPECT_NE(NetworkQualityEstimator::InvalidRTT(), rtt); | 112 EXPECT_NE(NetworkQualityEstimator::InvalidRTT(), rtt); |
146 EXPECT_NE(NetworkQualityEstimator::kInvalidThroughput, kbps); | 113 EXPECT_NE(NetworkQualityEstimator::kInvalidThroughput, kbps); |
147 | 114 |
148 EXPECT_NEAR( | 115 EXPECT_NEAR( |
149 rtt.InMilliseconds(), | 116 rtt.InMilliseconds(), |
150 estimator.GetRTTEstimateInternal(base::TimeTicks(), 100).InMilliseconds(), | 117 estimator.GetRTTEstimateInternal(base::TimeTicks(), 100).InMilliseconds(), |
151 1); | 118 1); |
152 | 119 |
153 // Check UMA histograms. | 120 // Check UMA histograms. |
154 histogram_tester.ExpectTotalCount("NQE.PeakKbps.Unknown", 0); | 121 histogram_tester.ExpectTotalCount("NQE.PeakKbps.Unknown", 0); |
155 histogram_tester.ExpectTotalCount("NQE.FastestRTT.Unknown", 0); | 122 histogram_tester.ExpectTotalCount("NQE.FastestRTT.Unknown", 0); |
156 | 123 |
157 histogram_tester.ExpectTotalCount("NQE.RatioEstimatedToActualRTT.Unknown", 0); | 124 histogram_tester.ExpectTotalCount("NQE.RatioEstimatedToActualRTT.Unknown", 0); |
158 | 125 |
159 scoped_ptr<URLRequest> request2(context.CreateRequest( | 126 estimator.NotifyHeadersReceived(*request); |
160 estimator.GetEchoURL(), DEFAULT_PRIORITY, &test_delegate)); | 127 estimator.NotifyRequestCompleted(*request); |
161 request2->SetLoadFlags(request2->load_flags() | LOAD_MAIN_FRAME); | |
162 request2->Start(); | |
163 base::RunLoop().Run(); | |
164 | |
165 histogram_tester.ExpectTotalCount("NQE.RTTObservations.Unknown", 1); | 128 histogram_tester.ExpectTotalCount("NQE.RTTObservations.Unknown", 1); |
166 estimator.SimulateNetworkChangeTo( | 129 estimator.SimulateNetworkChangeTo( |
167 NetworkChangeNotifier::ConnectionType::CONNECTION_WIFI, "test-1"); | 130 NetworkChangeNotifier::ConnectionType::CONNECTION_WIFI, "test-1"); |
168 histogram_tester.ExpectTotalCount("NQE.PeakKbps.Unknown", 1); | 131 histogram_tester.ExpectTotalCount("NQE.PeakKbps.Unknown", 1); |
169 histogram_tester.ExpectTotalCount("NQE.FastestRTT.Unknown", 1); | 132 histogram_tester.ExpectTotalCount("NQE.FastestRTT.Unknown", 1); |
170 | 133 |
171 histogram_tester.ExpectTotalCount("NQE.RatioMedianRTT.WiFi", 0); | 134 histogram_tester.ExpectTotalCount("NQE.RatioMedianRTT.WiFi", 0); |
172 | 135 |
173 histogram_tester.ExpectTotalCount("NQE.RTT.Percentile0.Unknown", 1); | 136 histogram_tester.ExpectTotalCount("NQE.RTT.Percentile0.Unknown", 1); |
174 histogram_tester.ExpectTotalCount("NQE.RTT.Percentile10.Unknown", 1); | 137 histogram_tester.ExpectTotalCount("NQE.RTT.Percentile10.Unknown", 1); |
(...skipping 21 matching lines...) Expand all Loading... |
196 estimator.GetRTTEstimateInternal(base::TimeTicks(), 100)); | 159 estimator.GetRTTEstimateInternal(base::TimeTicks(), 100)); |
197 EXPECT_EQ(NetworkQualityEstimator::kInvalidThroughput, | 160 EXPECT_EQ(NetworkQualityEstimator::kInvalidThroughput, |
198 estimator.GetDownlinkThroughputKbpsEstimateInternal( | 161 estimator.GetDownlinkThroughputKbpsEstimateInternal( |
199 base::TimeTicks(), 100)); | 162 base::TimeTicks(), 100)); |
200 | 163 |
201 EXPECT_FALSE(estimator.GetRTTEstimate(&rtt)); | 164 EXPECT_FALSE(estimator.GetRTTEstimate(&rtt)); |
202 EXPECT_FALSE(estimator.GetDownlinkThroughputKbpsEstimate(&kbps)); | 165 EXPECT_FALSE(estimator.GetDownlinkThroughputKbpsEstimate(&kbps)); |
203 } | 166 } |
204 | 167 |
205 TEST(NetworkQualityEstimatorTest, StoreObservations) { | 168 TEST(NetworkQualityEstimatorTest, StoreObservations) { |
| 169 net::test_server::EmbeddedTestServer embedded_test_server; |
| 170 embedded_test_server.ServeFilesFromDirectory( |
| 171 base::FilePath(FILE_PATH_LITERAL("net/data/url_request_unittest"))); |
| 172 ASSERT_TRUE(embedded_test_server.InitializeAndWaitUntilReady()); |
| 173 |
206 std::map<std::string, std::string> variation_params; | 174 std::map<std::string, std::string> variation_params; |
207 TestNetworkQualityEstimator estimator(variation_params); | 175 TestNetworkQualityEstimator estimator(variation_params); |
208 | |
209 TestDelegate test_delegate; | 176 TestDelegate test_delegate; |
210 TestURLRequestContext context(true); | 177 TestURLRequestContext context(false); |
211 context.set_network_quality_estimator(&estimator); | |
212 context.Init(); | |
213 | 178 |
214 // Push 10 more observations than the maximum buffer size. | 179 // Push 10 more observations than the maximum buffer size. |
215 for (size_t i = 0; i < estimator.kMaximumObservationsBufferSize + 10U; ++i) { | 180 for (size_t i = 0; i < estimator.kMaximumObservationsBufferSize + 10U; ++i) { |
216 scoped_ptr<URLRequest> request(context.CreateRequest( | 181 scoped_ptr<URLRequest> request( |
217 estimator.GetEchoURL(), DEFAULT_PRIORITY, &test_delegate)); | 182 context.CreateRequest(embedded_test_server.GetURL("/echo.html"), |
| 183 DEFAULT_PRIORITY, &test_delegate)); |
218 request->Start(); | 184 request->Start(); |
219 base::RunLoop().Run(); | 185 base::RunLoop().Run(); |
| 186 |
| 187 estimator.NotifyHeadersReceived(*request); |
| 188 estimator.NotifyRequestCompleted(*request); |
220 } | 189 } |
221 | 190 |
222 EXPECT_EQ(static_cast<size_t>( | 191 EXPECT_EQ(static_cast<size_t>( |
223 NetworkQualityEstimator::kMaximumObservationsBufferSize), | 192 NetworkQualityEstimator::kMaximumObservationsBufferSize), |
224 estimator.downstream_throughput_kbps_observations_.Size()); | 193 estimator.downstream_throughput_kbps_observations_.Size()); |
225 EXPECT_EQ(static_cast<size_t>( | 194 EXPECT_EQ(static_cast<size_t>( |
226 NetworkQualityEstimator::kMaximumObservationsBufferSize), | 195 NetworkQualityEstimator::kMaximumObservationsBufferSize), |
227 estimator.rtt_msec_observations_.Size()); | 196 estimator.rtt_msec_observations_.Size()); |
228 | 197 |
229 // Verify that the stored observations are cleared on network change. | 198 // Verify that the stored observations are cleared on network change. |
230 estimator.SimulateNetworkChangeTo( | 199 estimator.SimulateNetworkChangeTo( |
231 NetworkChangeNotifier::ConnectionType::CONNECTION_WIFI, "test-2"); | 200 NetworkChangeNotifier::ConnectionType::CONNECTION_WIFI, "test-2"); |
232 EXPECT_EQ(0U, estimator.downstream_throughput_kbps_observations_.Size()); | 201 EXPECT_EQ(0U, estimator.downstream_throughput_kbps_observations_.Size()); |
233 EXPECT_EQ(0U, estimator.rtt_msec_observations_.Size()); | 202 EXPECT_EQ(0U, estimator.rtt_msec_observations_.Size()); |
| 203 |
| 204 scoped_ptr<URLRequest> request( |
| 205 context.CreateRequest(embedded_test_server.GetURL("/echo.html"), |
| 206 DEFAULT_PRIORITY, &test_delegate)); |
234 } | 207 } |
235 #endif // !defined(OS_IOS) | |
236 | 208 |
237 // Verifies that the percentiles are correctly computed. All observations have | 209 // Verifies that the percentiles are correctly computed. All observations have |
238 // the same timestamp. Kbps percentiles must be in decreasing order. RTT | 210 // the same timestamp. Kbps percentiles must be in decreasing order. RTT |
239 // percentiles must be in increasing order. | 211 // percentiles must be in increasing order. |
240 TEST(NetworkQualityEstimatorTest, PercentileSameTimestamps) { | 212 TEST(NetworkQualityEstimatorTest, PercentileSameTimestamps) { |
241 std::map<std::string, std::string> variation_params; | 213 std::map<std::string, std::string> variation_params; |
242 TestNetworkQualityEstimator estimator(variation_params); | 214 TestNetworkQualityEstimator estimator(variation_params); |
243 base::TimeTicks now = base::TimeTicks::Now(); | 215 base::TimeTicks now = base::TimeTicks::Now(); |
244 | 216 |
245 // Network quality should be unavailable when no observations are available. | 217 // Network quality should be unavailable when no observations are available. |
(...skipping 90 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
336 estimator.GetRTTEstimateInternal( | 308 estimator.GetRTTEstimateInternal( |
337 base::TimeTicks::Now() + base::TimeDelta::FromMinutes(10), 50)); | 309 base::TimeTicks::Now() + base::TimeDelta::FromMinutes(10), 50)); |
338 EXPECT_EQ(NetworkQualityEstimator::kInvalidThroughput, | 310 EXPECT_EQ(NetworkQualityEstimator::kInvalidThroughput, |
339 estimator.GetDownlinkThroughputKbpsEstimateInternal( | 311 estimator.GetDownlinkThroughputKbpsEstimateInternal( |
340 base::TimeTicks::Now() + base::TimeDelta::FromMinutes(10), 50)); | 312 base::TimeTicks::Now() + base::TimeDelta::FromMinutes(10), 50)); |
341 } | 313 } |
342 | 314 |
343 // This test notifies NetworkQualityEstimator of received data. Next, | 315 // This test notifies NetworkQualityEstimator of received data. Next, |
344 // throughput and RTT percentiles are checked for correctness by doing simple | 316 // throughput and RTT percentiles are checked for correctness by doing simple |
345 // verifications. | 317 // verifications. |
346 #if !defined(OS_IOS) | |
347 // SpawnedTestServer is not supported on iOS. | |
348 TEST(NetworkQualityEstimatorTest, ComputedPercentiles) { | 318 TEST(NetworkQualityEstimatorTest, ComputedPercentiles) { |
| 319 net::test_server::EmbeddedTestServer embedded_test_server; |
| 320 embedded_test_server.ServeFilesFromDirectory( |
| 321 base::FilePath(FILE_PATH_LITERAL("net/data/url_request_unittest"))); |
| 322 ASSERT_TRUE(embedded_test_server.InitializeAndWaitUntilReady()); |
| 323 |
349 std::map<std::string, std::string> variation_params; | 324 std::map<std::string, std::string> variation_params; |
350 TestNetworkQualityEstimator estimator(variation_params); | 325 TestNetworkQualityEstimator estimator(variation_params); |
351 | 326 |
352 EXPECT_EQ(NetworkQualityEstimator::InvalidRTT(), | 327 EXPECT_EQ(NetworkQualityEstimator::InvalidRTT(), |
353 estimator.GetRTTEstimateInternal(base::TimeTicks(), 100)); | 328 estimator.GetRTTEstimateInternal(base::TimeTicks(), 100)); |
354 EXPECT_EQ(NetworkQualityEstimator::kInvalidThroughput, | 329 EXPECT_EQ(NetworkQualityEstimator::kInvalidThroughput, |
355 estimator.GetDownlinkThroughputKbpsEstimateInternal( | 330 estimator.GetDownlinkThroughputKbpsEstimateInternal( |
356 base::TimeTicks(), 100)); | 331 base::TimeTicks(), 100)); |
357 | 332 |
358 TestDelegate test_delegate; | 333 TestDelegate test_delegate; |
359 TestURLRequestContext context(true); | 334 TestURLRequestContext context(false); |
360 context.set_network_quality_estimator(&estimator); | |
361 context.Init(); | |
362 | 335 |
363 // Number of observations are more than the maximum buffer size. | 336 // Number of observations are more than the maximum buffer size. |
364 for (size_t i = 0; i < estimator.kMaximumObservationsBufferSize + 100U; ++i) { | 337 for (size_t i = 0; i < estimator.kMaximumObservationsBufferSize + 100U; ++i) { |
365 scoped_ptr<URLRequest> request(context.CreateRequest( | 338 scoped_ptr<URLRequest> request( |
366 estimator.GetEchoURL(), DEFAULT_PRIORITY, &test_delegate)); | 339 context.CreateRequest(embedded_test_server.GetURL("/echo.html"), |
| 340 DEFAULT_PRIORITY, &test_delegate)); |
367 request->Start(); | 341 request->Start(); |
368 base::RunLoop().Run(); | 342 base::RunLoop().Run(); |
| 343 |
| 344 // Use different number of bytes to create variation. |
| 345 estimator.NotifyHeadersReceived(*request); |
| 346 estimator.NotifyRequestCompleted(*request); |
369 } | 347 } |
370 | 348 |
371 // Verify the percentiles through simple tests. | 349 // Verify the percentiles through simple tests. |
372 for (int i = 0; i <= 100; ++i) { | 350 for (int i = 0; i <= 100; ++i) { |
373 EXPECT_GT(estimator.GetDownlinkThroughputKbpsEstimateInternal( | 351 EXPECT_GT(estimator.GetDownlinkThroughputKbpsEstimateInternal( |
374 base::TimeTicks(), i), | 352 base::TimeTicks(), i), |
375 0); | 353 0); |
376 EXPECT_LT(estimator.GetRTTEstimateInternal(base::TimeTicks(), i), | 354 EXPECT_LT(estimator.GetRTTEstimateInternal(base::TimeTicks(), i), |
377 base::TimeDelta::Max()); | 355 base::TimeDelta::Max()); |
378 | 356 |
379 if (i != 0) { | 357 if (i != 0) { |
380 // Throughput percentiles are in decreasing order. | 358 // Throughput percentiles are in decreasing order. |
381 EXPECT_LE(estimator.GetDownlinkThroughputKbpsEstimateInternal( | 359 EXPECT_LE(estimator.GetDownlinkThroughputKbpsEstimateInternal( |
382 base::TimeTicks(), i), | 360 base::TimeTicks(), i), |
383 estimator.GetDownlinkThroughputKbpsEstimateInternal( | 361 estimator.GetDownlinkThroughputKbpsEstimateInternal( |
384 base::TimeTicks(), i - 1)); | 362 base::TimeTicks(), i - 1)); |
385 | 363 |
386 // RTT percentiles are in increasing order. | 364 // RTT percentiles are in increasing order. |
387 EXPECT_GE(estimator.GetRTTEstimateInternal(base::TimeTicks(), i), | 365 EXPECT_GE(estimator.GetRTTEstimateInternal(base::TimeTicks(), i), |
388 estimator.GetRTTEstimateInternal(base::TimeTicks(), i - 1)); | 366 estimator.GetRTTEstimateInternal(base::TimeTicks(), i - 1)); |
389 } | 367 } |
390 } | 368 } |
391 } | 369 } |
392 #endif // !defined(OS_IOS) | |
393 | 370 |
394 TEST(NetworkQualityEstimatorTest, ObtainOperatingParams) { | 371 TEST(NetworkQualityEstimatorTest, ObtainOperatingParams) { |
395 std::map<std::string, std::string> variation_params; | 372 std::map<std::string, std::string> variation_params; |
396 variation_params["Unknown.DefaultMedianKbps"] = "100"; | 373 variation_params["Unknown.DefaultMedianKbps"] = "100"; |
397 variation_params["WiFi.DefaultMedianKbps"] = "200"; | 374 variation_params["WiFi.DefaultMedianKbps"] = "200"; |
398 variation_params["2G.DefaultMedianKbps"] = "300"; | 375 variation_params["2G.DefaultMedianKbps"] = "300"; |
399 | 376 |
400 variation_params["Unknown.DefaultMedianRTTMsec"] = "1000"; | 377 variation_params["Unknown.DefaultMedianRTTMsec"] = "1000"; |
401 variation_params["WiFi.DefaultMedianRTTMsec"] = "2000"; | 378 variation_params["WiFi.DefaultMedianRTTMsec"] = "2000"; |
402 // Negative variation value should not be used. | 379 // Negative variation value should not be used. |
(...skipping 267 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
670 EXPECT_EQ(100, rtt.InMilliseconds()); | 647 EXPECT_EQ(100, rtt.InMilliseconds()); |
671 | 648 |
672 int32_t downstream_throughput_kbps; | 649 int32_t downstream_throughput_kbps; |
673 EXPECT_FALSE(estimator.GetRecentMedianDownlinkThroughputKbps( | 650 EXPECT_FALSE(estimator.GetRecentMedianDownlinkThroughputKbps( |
674 now + base::TimeDelta::FromSeconds(10), &downstream_throughput_kbps)); | 651 now + base::TimeDelta::FromSeconds(10), &downstream_throughput_kbps)); |
675 EXPECT_TRUE(estimator.GetRecentMedianDownlinkThroughputKbps( | 652 EXPECT_TRUE(estimator.GetRecentMedianDownlinkThroughputKbps( |
676 now, &downstream_throughput_kbps)); | 653 now, &downstream_throughput_kbps)); |
677 EXPECT_EQ(100, downstream_throughput_kbps); | 654 EXPECT_EQ(100, downstream_throughput_kbps); |
678 } | 655 } |
679 | 656 |
680 // An external estimate provider that does not have a valid RTT or throughput | |
681 // estimate. | |
682 class InvalidExternalEstimateProvider : public ExternalEstimateProvider { | |
683 public: | |
684 InvalidExternalEstimateProvider() : get_rtt_count_(0) {} | |
685 ~InvalidExternalEstimateProvider() override {} | |
686 | |
687 // ExternalEstimateProvider implementation: | |
688 bool GetRTT(base::TimeDelta* rtt) const override { | |
689 DCHECK(rtt); | |
690 get_rtt_count_++; | |
691 return false; | |
692 } | |
693 | |
694 // ExternalEstimateProvider implementation: | |
695 bool GetDownstreamThroughputKbps( | |
696 int32_t* downstream_throughput_kbps) const override { | |
697 DCHECK(downstream_throughput_kbps); | |
698 return false; | |
699 } | |
700 | |
701 // ExternalEstimateProvider implementation: | |
702 bool GetUpstreamThroughputKbps( | |
703 int32_t* upstream_throughput_kbps) const override { | |
704 // NetworkQualityEstimator does not support upstream throughput. | |
705 ADD_FAILURE(); | |
706 return false; | |
707 } | |
708 | |
709 // ExternalEstimateProvider implementation: | |
710 bool GetTimeSinceLastUpdate( | |
711 base::TimeDelta* time_since_last_update) const override { | |
712 *time_since_last_update = base::TimeDelta::FromMilliseconds(1); | |
713 return true; | |
714 } | |
715 | |
716 // ExternalEstimateProvider implementation: | |
717 void SetUpdatedEstimateDelegate(UpdatedEstimateDelegate* delegate) override {} | |
718 | |
719 // ExternalEstimateProvider implementation: | |
720 void Update() const override {} | |
721 | |
722 size_t get_rtt_count() const { return get_rtt_count_; } | |
723 | |
724 private: | |
725 // Keeps track of number of times different functions were called. | |
726 mutable size_t get_rtt_count_; | |
727 | |
728 DISALLOW_COPY_AND_ASSIGN(InvalidExternalEstimateProvider); | |
729 }; | |
730 | |
731 // Tests if the RTT value from external estimate provider is discarded if the | |
732 // external estimate provider is invalid. | |
733 TEST(NetworkQualityEstimatorTest, InvalidExternalEstimateProvider) { | |
734 InvalidExternalEstimateProvider* invalid_external_estimate_provider = | |
735 new InvalidExternalEstimateProvider(); | |
736 scoped_ptr<ExternalEstimateProvider> external_estimate_provider( | |
737 invalid_external_estimate_provider); | |
738 | |
739 TestNetworkQualityEstimator estimator(std::map<std::string, std::string>(), | |
740 external_estimate_provider.Pass()); | |
741 | |
742 base::TimeDelta rtt; | |
743 int32_t kbps; | |
744 EXPECT_EQ(1U, invalid_external_estimate_provider->get_rtt_count()); | |
745 EXPECT_FALSE(estimator.GetRTTEstimate(&rtt)); | |
746 EXPECT_FALSE(estimator.GetDownlinkThroughputKbpsEstimate(&kbps)); | |
747 } | |
748 | |
749 class TestExternalEstimateProvider : public ExternalEstimateProvider { | |
750 public: | |
751 TestExternalEstimateProvider(base::TimeDelta rtt, | |
752 int32_t downstream_throughput_kbps) | |
753 : rtt_(rtt), | |
754 downstream_throughput_kbps_(downstream_throughput_kbps), | |
755 time_since_last_update_(base::TimeDelta::FromSeconds(1)), | |
756 get_time_since_last_update_count_(0), | |
757 get_rtt_count_(0), | |
758 get_downstream_throughput_kbps_count_(0), | |
759 update_count_(0) {} | |
760 ~TestExternalEstimateProvider() override {} | |
761 | |
762 // ExternalEstimateProvider implementation: | |
763 bool GetRTT(base::TimeDelta* rtt) const override { | |
764 *rtt = rtt_; | |
765 get_rtt_count_++; | |
766 return true; | |
767 } | |
768 | |
769 // ExternalEstimateProvider implementation: | |
770 bool GetDownstreamThroughputKbps( | |
771 int32_t* downstream_throughput_kbps) const override { | |
772 *downstream_throughput_kbps = downstream_throughput_kbps_; | |
773 get_downstream_throughput_kbps_count_++; | |
774 return true; | |
775 } | |
776 | |
777 // ExternalEstimateProvider implementation: | |
778 bool GetUpstreamThroughputKbps( | |
779 int32_t* upstream_throughput_kbps) const override { | |
780 // NetworkQualityEstimator does not support upstream throughput. | |
781 ADD_FAILURE(); | |
782 return false; | |
783 } | |
784 | |
785 // ExternalEstimateProvider implementation: | |
786 bool GetTimeSinceLastUpdate( | |
787 base::TimeDelta* time_since_last_update) const override { | |
788 *time_since_last_update = time_since_last_update_; | |
789 get_time_since_last_update_count_++; | |
790 return true; | |
791 } | |
792 | |
793 // ExternalEstimateProvider implementation: | |
794 void SetUpdatedEstimateDelegate(UpdatedEstimateDelegate* delegate) override {} | |
795 | |
796 // ExternalEstimateProvider implementation: | |
797 void Update() const override { update_count_++; } | |
798 | |
799 void set_time_since_last_update(base::TimeDelta time_since_last_update) { | |
800 time_since_last_update_ = time_since_last_update; | |
801 } | |
802 | |
803 size_t get_time_since_last_update_count() const { | |
804 return get_time_since_last_update_count_; | |
805 } | |
806 size_t get_rtt_count() const { return get_rtt_count_; } | |
807 size_t get_downstream_throughput_kbps_count() const { | |
808 return get_downstream_throughput_kbps_count_; | |
809 } | |
810 size_t update_count() const { return update_count_; } | |
811 | |
812 private: | |
813 // RTT and downstream throughput estimates. | |
814 const base::TimeDelta rtt_; | |
815 const int32_t downstream_throughput_kbps_; | |
816 | |
817 base::TimeDelta time_since_last_update_; | |
818 | |
819 // Keeps track of number of times different functions were called. | |
820 mutable size_t get_time_since_last_update_count_; | |
821 mutable size_t get_rtt_count_; | |
822 mutable size_t get_downstream_throughput_kbps_count_; | |
823 mutable size_t update_count_; | |
824 | |
825 DISALLOW_COPY_AND_ASSIGN(TestExternalEstimateProvider); | |
826 }; | |
827 | |
828 // Tests if the external estimate provider is called in the constructor and | |
829 // on network change notification. | |
830 TEST(NetworkQualityEstimatorTest, TestExternalEstimateProvider) { | |
831 TestExternalEstimateProvider* test_external_estimate_provider = | |
832 new TestExternalEstimateProvider(base::TimeDelta::FromMilliseconds(1), | |
833 100); | |
834 scoped_ptr<ExternalEstimateProvider> external_estimate_provider( | |
835 test_external_estimate_provider); | |
836 std::map<std::string, std::string> variation_params; | |
837 TestNetworkQualityEstimator estimator(variation_params, | |
838 external_estimate_provider.Pass()); | |
839 | |
840 base::TimeDelta rtt; | |
841 int32_t kbps; | |
842 EXPECT_TRUE(estimator.GetRTTEstimate(&rtt)); | |
843 EXPECT_TRUE(estimator.GetDownlinkThroughputKbpsEstimate(&kbps)); | |
844 | |
845 EXPECT_EQ( | |
846 1U, test_external_estimate_provider->get_time_since_last_update_count()); | |
847 EXPECT_EQ(1U, test_external_estimate_provider->get_rtt_count()); | |
848 EXPECT_EQ( | |
849 1U, | |
850 test_external_estimate_provider->get_downstream_throughput_kbps_count()); | |
851 | |
852 // Change network type to WiFi. Number of queries to External estimate | |
853 // provider must increment. | |
854 estimator.SimulateNetworkChangeTo( | |
855 NetworkChangeNotifier::ConnectionType::CONNECTION_WIFI, "test-1"); | |
856 EXPECT_TRUE(estimator.GetRTTEstimate(&rtt)); | |
857 EXPECT_TRUE(estimator.GetDownlinkThroughputKbpsEstimate(&kbps)); | |
858 EXPECT_EQ( | |
859 2U, test_external_estimate_provider->get_time_since_last_update_count()); | |
860 EXPECT_EQ(2U, test_external_estimate_provider->get_rtt_count()); | |
861 EXPECT_EQ( | |
862 2U, | |
863 test_external_estimate_provider->get_downstream_throughput_kbps_count()); | |
864 | |
865 // Change network type to 2G. Number of queries to External estimate provider | |
866 // must increment. | |
867 estimator.SimulateNetworkChangeTo( | |
868 NetworkChangeNotifier::ConnectionType::CONNECTION_2G, "test-1"); | |
869 EXPECT_EQ( | |
870 3U, test_external_estimate_provider->get_time_since_last_update_count()); | |
871 EXPECT_EQ(3U, test_external_estimate_provider->get_rtt_count()); | |
872 EXPECT_EQ( | |
873 3U, | |
874 test_external_estimate_provider->get_downstream_throughput_kbps_count()); | |
875 | |
876 // Set the external estimate as old. Network Quality estimator should request | |
877 // an update on connection type change. | |
878 EXPECT_EQ(0U, test_external_estimate_provider->update_count()); | |
879 test_external_estimate_provider->set_time_since_last_update( | |
880 base::TimeDelta::Max()); | |
881 | |
882 estimator.SimulateNetworkChangeTo( | |
883 NetworkChangeNotifier::ConnectionType::CONNECTION_2G, "test-2"); | |
884 EXPECT_EQ( | |
885 4U, test_external_estimate_provider->get_time_since_last_update_count()); | |
886 EXPECT_EQ(3U, test_external_estimate_provider->get_rtt_count()); | |
887 EXPECT_EQ( | |
888 3U, | |
889 test_external_estimate_provider->get_downstream_throughput_kbps_count()); | |
890 EXPECT_EQ(1U, test_external_estimate_provider->update_count()); | |
891 | |
892 // Estimates are unavailable because external estimate provider never | |
893 // notifies network quality estimator of the updated estimates. | |
894 EXPECT_FALSE(estimator.GetRTTEstimate(&rtt)); | |
895 EXPECT_FALSE(estimator.GetDownlinkThroughputKbpsEstimate(&kbps)); | |
896 } | |
897 | |
898 // Tests if the estimate from the external estimate provider is merged with the | |
899 // observations collected from the HTTP requests. | |
900 #if !defined(OS_IOS) | |
901 // SpawnedTestServer is not supported on iOS. | |
902 TEST(NetworkQualityEstimatorTest, TestExternalEstimateProviderMergeEstimates) { | |
903 const base::TimeDelta external_estimate_provider_rtt = | |
904 base::TimeDelta::FromMilliseconds(1); | |
905 const int32_t external_estimate_provider_downstream_throughput = 100; | |
906 TestExternalEstimateProvider* test_external_estimate_provider = | |
907 new TestExternalEstimateProvider( | |
908 external_estimate_provider_rtt, | |
909 external_estimate_provider_downstream_throughput); | |
910 scoped_ptr<ExternalEstimateProvider> external_estimate_provider( | |
911 test_external_estimate_provider); | |
912 | |
913 std::map<std::string, std::string> variation_params; | |
914 TestNetworkQualityEstimator estimator(variation_params, | |
915 external_estimate_provider.Pass()); | |
916 | |
917 base::TimeDelta rtt; | |
918 // Estimate provided by network quality estimator should match the estimate | |
919 // provided by external estimate provider. | |
920 EXPECT_TRUE(estimator.GetRTTEstimate(&rtt)); | |
921 EXPECT_EQ(external_estimate_provider_rtt, rtt); | |
922 | |
923 int32_t kbps; | |
924 EXPECT_TRUE(estimator.GetDownlinkThroughputKbpsEstimate(&kbps)); | |
925 EXPECT_EQ(external_estimate_provider_downstream_throughput, kbps); | |
926 | |
927 EXPECT_EQ(1U, estimator.rtt_msec_observations_.Size()); | |
928 EXPECT_EQ(1U, estimator.downstream_throughput_kbps_observations_.Size()); | |
929 | |
930 TestDelegate test_delegate; | |
931 TestURLRequestContext context(true); | |
932 context.set_network_quality_estimator(&estimator); | |
933 context.Init(); | |
934 | |
935 scoped_ptr<URLRequest> request(context.CreateRequest( | |
936 estimator.GetEchoURL(), DEFAULT_PRIORITY, &test_delegate)); | |
937 request->Start(); | |
938 base::RunLoop().Run(); | |
939 | |
940 EXPECT_EQ(2U, estimator.rtt_msec_observations_.Size()); | |
941 EXPECT_EQ(2U, estimator.downstream_throughput_kbps_observations_.Size()); | |
942 } | |
943 #endif // !defined(OS_IOS) | |
944 | |
945 } // namespace net | 657 } // namespace net |
OLD | NEW |