forked from chromium/chromium
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnetwork_tasks_unittest.cc
250 lines (210 loc) · 8.97 KB
/
network_tasks_unittest.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
// Copyright 2022 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 "components/cronet/cronet_context.h"
#include <latch>
#include "base/test/bind.h"
#include "base/test/task_environment.h"
#include "components/cronet/cronet_global_state.h"
#include "components/cronet/url_request_context_config.h"
#include "net/base/mock_network_change_notifier.h"
#include "net/base/request_priority.h"
#include "net/cert/cert_verifier.h"
#include "net/proxy_resolution/proxy_config_service_fixed.h"
#include "net/traffic_annotation/network_traffic_annotation_test_helper.h"
#include "net/url_request/url_request.h"
#include "net/url_request/url_request_context.h"
#include "testing/gtest/include/gtest/gtest.h"
#if BUILDFLAG(IS_ANDROID)
#include "base/android/build_info.h"
#endif // BUILDFLAG(IS_ANDROID)
namespace cronet {
namespace {
class NoOpCronetContextCallback : public CronetContext::Callback {
public:
NoOpCronetContextCallback() = default;
NoOpCronetContextCallback(const NoOpCronetContextCallback&) = delete;
NoOpCronetContextCallback& operator=(const NoOpCronetContextCallback&) =
delete;
void OnInitNetworkThread() override {}
void OnDestroyNetworkThread() override {}
void OnEffectiveConnectionTypeChanged(
net::EffectiveConnectionType effective_connection_type) override {}
void OnRTTOrThroughputEstimatesComputed(
int32_t http_rtt_ms,
int32_t transport_rtt_ms,
int32_t downstream_throughput_kbps) override {}
void OnRTTObservation(int32_t rtt_ms,
int32_t timestamp_ms,
net::NetworkQualityObservationSource source) override {}
void OnThroughputObservation(
int32_t throughput_kbps,
int32_t timestamp_ms,
net::NetworkQualityObservationSource source) override {}
void OnStopNetLogCompleted() override {}
~NoOpCronetContextCallback() override = default;
};
std::unique_ptr<URLRequestContextConfig> CreateSimpleURLRequestContextConfig() {
return URLRequestContextConfig::CreateURLRequestContextConfig(
// Enable QUIC.
true,
// QUIC User Agent ID.
"Default QUIC User Agent ID",
// Enable SPDY.
true,
// Enable Brotli.
false,
// Type of http cache.
URLRequestContextConfig::HttpCacheType::DISK,
// Max size of http cache in bytes.
1024000,
// Disable caching for HTTP responses. Other information may be stored
// in the cache.
false,
// Storage path for http cache and cookie storage.
"/data/data/org.chromium.net/app_cronet_test/test_storage",
// Accept-Language request header field.
"foreign-language",
// User-Agent request header field.
"fake agent",
// JSON encoded experimental options.
"",
// MockCertVerifier to use for testing purposes.
std::unique_ptr<net::CertVerifier>(),
// Enable network quality estimator.
false,
// Enable Public Key Pinning bypass for local trust anchors.
true,
// Optional network thread priority.
absl::nullopt);
}
class NetworkTasksTest : public testing::Test {
protected:
NetworkTasksTest()
: ncn_(net::NetworkChangeNotifier::CreateMockIfNeeded()),
scoped_ncn_(
std::make_unique<net::test::ScopedMockNetworkChangeNotifier>()),
network_thread_(std::make_unique<base::Thread>("network")),
file_thread_(std::make_unique<base::Thread>("Network File Thread")),
network_tasks_(new CronetContext::NetworkTasks(
CreateSimpleURLRequestContextConfig(),
std::make_unique<NoOpCronetContextCallback>())) {
base::Thread::Options options;
options.message_pump_type = base::MessagePumpType::IO;
network_thread_->StartWithOptions(std::move(options));
network_task_runner_ = network_thread_->task_runner();
file_thread_->Start();
file_task_runner_ = file_thread_->task_runner();
scoped_ncn_->mock_network_change_notifier()->ForceNetworkHandlesSupported();
Initialize();
}
~NetworkTasksTest() override {
PostToNetworkThreadSync(
base::BindLambdaForTesting([&]() { delete network_tasks_; }));
}
void Initialize() {
PostToNetworkThreadSync(
base::BindOnce(&CronetContext::NetworkTasks::Initialize,
base::Unretained(network_tasks_), network_task_runner_,
file_task_runner_,
std::make_unique<net::ProxyConfigServiceFixed>(
net::ProxyConfigWithAnnotation::CreateDirect())));
}
void SpawnNetworkBoundURLRequestContext(net::handles::NetworkHandle network) {
PostToNetworkThreadSync(base::BindLambdaForTesting([=]() {
network_tasks_->SpawnNetworkBoundURLRequestContextForTesting(network);
}));
}
void CheckURLRequestContextExistence(net::handles::NetworkHandle network,
bool expected) {
std::atomic_bool context_exists = false;
PostToNetworkThreadSync(base::BindLambdaForTesting([&]() {
context_exists.store(
network_tasks_->URLRequestContextExistsForTesting(network));
}));
EXPECT_EQ(expected, context_exists.load());
}
void CreateURLRequest(net::handles::NetworkHandle network) {
std::atomic_bool url_request_created = false;
PostToNetworkThreadSync(base::BindLambdaForTesting([&]() {
auto* context = network_tasks_->GetURLRequestContext(network);
url_request_ = context->CreateRequest(GURL("http://www.foo.com"),
net::DEFAULT_PRIORITY, nullptr,
TRAFFIC_ANNOTATION_FOR_TESTS);
url_request_created = !!url_request_;
}));
EXPECT_TRUE(url_request_created);
}
void ReleaseURLRequest() {
PostToNetworkThreadSync(
base::BindLambdaForTesting([&]() { url_request_.reset(); }));
}
void MaybeDestroyURLRequestContext(net::handles::NetworkHandle network) {
PostToNetworkThreadSync(base::BindLambdaForTesting(
[&]() { network_tasks_->MaybeDestroyURLRequestContext(network); }));
}
void PostToNetworkThreadSync(base::OnceCallback<void()> callback) {
std::latch callback_executed{1};
auto wait_for_callback = base::BindLambdaForTesting(
[&callback_executed]() { callback_executed.count_down(); });
network_task_runner_->PostTask(
FROM_HERE, std::move(callback).Then(std::move(wait_for_callback)));
callback_executed.wait();
}
base::test::TaskEnvironment task_environment_;
std::unique_ptr<net::NetworkChangeNotifier> ncn_;
std::unique_ptr<net::test::ScopedMockNetworkChangeNotifier> scoped_ncn_;
std::unique_ptr<base::Thread> network_thread_;
std::unique_ptr<base::Thread> file_thread_;
scoped_refptr<base::SingleThreadTaskRunner> network_task_runner_;
scoped_refptr<base::SingleThreadTaskRunner> file_task_runner_;
raw_ptr<CronetContext::NetworkTasks> network_tasks_;
std::unique_ptr<net::URLRequest> url_request_;
};
TEST_F(NetworkTasksTest, NetworkBoundContextLifetime) {
#if BUILDFLAG(IS_ANDROID)
if (base::android::BuildInfo::GetInstance()->sdk_int() <
base::android::SDK_VERSION_MARSHMALLOW) {
GTEST_SKIP() << "Network binding on Android requires an API level >= 23";
}
constexpr net::handles::NetworkHandle kNetwork = 1;
CheckURLRequestContextExistence(kNetwork, false);
SpawnNetworkBoundURLRequestContext(kNetwork);
CheckURLRequestContextExistence(kNetwork, true);
// Once the network disconnects the context should be destroyed.
scoped_ncn_->mock_network_change_notifier()->NotifyNetworkDisconnected(
kNetwork);
CheckURLRequestContextExistence(kNetwork, false);
#else
GTEST_SKIP() << "Network binding is supported only on Android";
#endif // BUILDFLAG(IS_ANDROID)
}
TEST_F(NetworkTasksTest, NetworkBoundContextWithPendingRequest) {
#if BUILDFLAG(IS_ANDROID)
if (base::android::BuildInfo::GetInstance()->sdk_int() <
base::android::SDK_VERSION_MARSHMALLOW) {
GTEST_SKIP() << "Network binding on Android requires an API level >= 23";
}
constexpr net::handles::NetworkHandle kNetwork = 1;
CheckURLRequestContextExistence(kNetwork, false);
SpawnNetworkBoundURLRequestContext(kNetwork);
CheckURLRequestContextExistence(kNetwork, true);
// If after a network disconnection there are still pending requests, the
// context should not be destroyed to avoid UAFs (URLRequests can reference
// their associated URLRequestContext).
CreateURLRequest(kNetwork);
CheckURLRequestContextExistence(kNetwork, true);
scoped_ncn_->mock_network_change_notifier()->QueueNetworkDisconnected(
kNetwork);
CheckURLRequestContextExistence(kNetwork, true);
// Once the URLRequest is destroyed, MaybeDestroyURLRequestContext should be
// able to destroy the context.
ReleaseURLRequest();
MaybeDestroyURLRequestContext(kNetwork);
CheckURLRequestContextExistence(kNetwork, false);
#else
GTEST_SKIP() << "Network binding is supported only on Android";
#endif // BUILDFLAG(IS_ANDROID)
}
} // namespace
} // namespace cronet