1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "content/browser/resolve_proxy_msg_helper.h"
7 #include "content/browser/browser_thread_impl.h"
8 #include "content/common/view_messages.h"
9 #include "ipc/ipc_test_sink.h"
10 #include "net/base/net_errors.h"
11 #include "net/proxy/mock_proxy_resolver.h"
12 #include "net/proxy/proxy_config_service.h"
13 #include "net/proxy/proxy_service.h"
14 #include "testing/gtest/include/gtest/gtest.h"
18 // This ProxyConfigService always returns "http://pac" as the PAC url to use.
19 class MockProxyConfigService
: public net::ProxyConfigService
{
21 void AddObserver(Observer
* observer
) override
{}
22 void RemoveObserver(Observer
* observer
) override
{}
23 ConfigAvailability
GetLatestProxyConfig(net::ProxyConfig
* results
) override
{
24 *results
= net::ProxyConfig::CreateFromCustomPacURL(GURL("http://pac"));
29 class TestResolveProxyMsgHelper
: public ResolveProxyMsgHelper
{
31 TestResolveProxyMsgHelper(
32 net::ProxyService
* proxy_service
,
33 IPC::Listener
* listener
)
34 : ResolveProxyMsgHelper(proxy_service
),
35 listener_(listener
) {}
36 bool Send(IPC::Message
* message
) override
{
37 listener_
->OnMessageReceived(*message
);
43 ~TestResolveProxyMsgHelper() override
{}
45 IPC::Listener
* listener_
;
48 class ResolveProxyMsgHelperTest
: public testing::Test
, public IPC::Listener
{
50 struct PendingResult
{
51 PendingResult(bool result
,
52 const std::string
& proxy_list
)
53 : result(result
), proxy_list(proxy_list
) {
57 std::string proxy_list
;
60 ResolveProxyMsgHelperTest()
61 : resolver_factory_(new net::MockAsyncProxyResolverFactory(false)),
62 service_(new net::ProxyService(new MockProxyConfigService
,
63 make_scoped_ptr(resolver_factory_
),
65 helper_(new TestResolveProxyMsgHelper(service_
.get(), this)),
66 io_thread_(BrowserThread::IO
, &message_loop_
) {
67 test_sink_
.AddFilter(this);
71 const PendingResult
* pending_result() const { return pending_result_
.get(); }
73 void clear_pending_result() {
74 pending_result_
.reset();
77 IPC::Message
* GenerateReply() {
79 std::string temp_string
;
80 ViewHostMsg_ResolveProxy
message(GURL(), &temp_bool
, &temp_string
);
81 return IPC::SyncMessage::GenerateReply(&message
);
84 net::MockAsyncProxyResolverFactory
* resolver_factory_
;
85 net::MockAsyncProxyResolver resolver_
;
86 scoped_ptr
<net::ProxyService
> service_
;
87 scoped_refptr
<ResolveProxyMsgHelper
> helper_
;
88 scoped_ptr
<PendingResult
> pending_result_
;
91 bool OnMessageReceived(const IPC::Message
& msg
) override
{
92 TupleTypes
<ViewHostMsg_ResolveProxy::ReplyParam
>::ValueTuple reply_data
;
93 EXPECT_TRUE(ViewHostMsg_ResolveProxy::ReadReplyParam(&msg
, &reply_data
));
94 DCHECK(!pending_result_
.get());
95 pending_result_
.reset(
96 new PendingResult(get
<0>(reply_data
), get
<1>(reply_data
)));
97 test_sink_
.ClearMessages();
101 base::MessageLoopForIO message_loop_
;
102 BrowserThreadImpl io_thread_
;
103 IPC::TestSink test_sink_
;
106 // Issue three sequential requests -- each should succeed.
107 TEST_F(ResolveProxyMsgHelperTest
, Sequential
) {
108 GURL
url1("http://www.google1.com/");
109 GURL
url2("http://www.google2.com/");
110 GURL
url3("http://www.google3.com/");
112 // Messages are deleted by the sink.
113 IPC::Message
* msg1
= GenerateReply();
114 IPC::Message
* msg2
= GenerateReply();
115 IPC::Message
* msg3
= GenerateReply();
117 // Execute each request sequentially (so there are never 2 requests
118 // outstanding at the same time).
120 helper_
->OnResolveProxy(url1
, msg1
);
122 // Finish ProxyService's initialization.
123 ASSERT_EQ(1u, resolver_factory_
->pending_requests().size());
124 resolver_factory_
->pending_requests()[0]->CompleteNowWithForwarder(
125 net::OK
, &resolver_
);
127 ASSERT_EQ(1u, resolver_
.pending_requests().size());
128 EXPECT_EQ(url1
, resolver_
.pending_requests()[0]->url());
129 resolver_
.pending_requests()[0]->results()->UseNamedProxy("result1:80");
130 resolver_
.pending_requests()[0]->CompleteNow(net::OK
);
133 EXPECT_EQ(true, pending_result()->result
);
134 EXPECT_EQ("PROXY result1:80", pending_result()->proxy_list
);
135 clear_pending_result();
137 helper_
->OnResolveProxy(url2
, msg2
);
139 ASSERT_EQ(1u, resolver_
.pending_requests().size());
140 EXPECT_EQ(url2
, resolver_
.pending_requests()[0]->url());
141 resolver_
.pending_requests()[0]->results()->UseNamedProxy("result2:80");
142 resolver_
.pending_requests()[0]->CompleteNow(net::OK
);
145 EXPECT_EQ(true, pending_result()->result
);
146 EXPECT_EQ("PROXY result2:80", pending_result()->proxy_list
);
147 clear_pending_result();
149 helper_
->OnResolveProxy(url3
, msg3
);
151 ASSERT_EQ(1u, resolver_
.pending_requests().size());
152 EXPECT_EQ(url3
, resolver_
.pending_requests()[0]->url());
153 resolver_
.pending_requests()[0]->results()->UseNamedProxy("result3:80");
154 resolver_
.pending_requests()[0]->CompleteNow(net::OK
);
157 EXPECT_EQ(true, pending_result()->result
);
158 EXPECT_EQ("PROXY result3:80", pending_result()->proxy_list
);
159 clear_pending_result();
162 // Issue a request while one is already in progress -- should be queued.
163 TEST_F(ResolveProxyMsgHelperTest
, QueueRequests
) {
164 GURL
url1("http://www.google1.com/");
165 GURL
url2("http://www.google2.com/");
166 GURL
url3("http://www.google3.com/");
168 IPC::Message
* msg1
= GenerateReply();
169 IPC::Message
* msg2
= GenerateReply();
170 IPC::Message
* msg3
= GenerateReply();
172 // Start three requests. Since the proxy resolver is async, all the
173 // requests will be pending.
175 helper_
->OnResolveProxy(url1
, msg1
);
177 // Finish ProxyService's initialization.
178 ASSERT_EQ(1u, resolver_factory_
->pending_requests().size());
179 resolver_factory_
->pending_requests()[0]->CompleteNowWithForwarder(
180 net::OK
, &resolver_
);
182 helper_
->OnResolveProxy(url2
, msg2
);
183 helper_
->OnResolveProxy(url3
, msg3
);
185 // ResolveProxyHelper only keeps 1 request outstanding in ProxyService
187 ASSERT_EQ(1u, resolver_
.pending_requests().size());
188 EXPECT_EQ(url1
, resolver_
.pending_requests()[0]->url());
190 resolver_
.pending_requests()[0]->results()->UseNamedProxy("result1:80");
191 resolver_
.pending_requests()[0]->CompleteNow(net::OK
);
194 EXPECT_EQ(true, pending_result()->result
);
195 EXPECT_EQ("PROXY result1:80", pending_result()->proxy_list
);
196 clear_pending_result();
198 ASSERT_EQ(1u, resolver_
.pending_requests().size());
199 EXPECT_EQ(url2
, resolver_
.pending_requests()[0]->url());
201 resolver_
.pending_requests()[0]->results()->UseNamedProxy("result2:80");
202 resolver_
.pending_requests()[0]->CompleteNow(net::OK
);
205 EXPECT_EQ(true, pending_result()->result
);
206 EXPECT_EQ("PROXY result2:80", pending_result()->proxy_list
);
207 clear_pending_result();
209 ASSERT_EQ(1u, resolver_
.pending_requests().size());
210 EXPECT_EQ(url3
, resolver_
.pending_requests()[0]->url());
212 resolver_
.pending_requests()[0]->results()->UseNamedProxy("result3:80");
213 resolver_
.pending_requests()[0]->CompleteNow(net::OK
);
216 EXPECT_EQ(true, pending_result()->result
);
217 EXPECT_EQ("PROXY result3:80", pending_result()->proxy_list
);
218 clear_pending_result();
221 // Delete the helper while a request is in progress, and others are pending.
222 TEST_F(ResolveProxyMsgHelperTest
, CancelPendingRequests
) {
223 GURL
url1("http://www.google1.com/");
224 GURL
url2("http://www.google2.com/");
225 GURL
url3("http://www.google3.com/");
227 // They will be deleted by the request's cancellation.
228 IPC::Message
* msg1
= GenerateReply();
229 IPC::Message
* msg2
= GenerateReply();
230 IPC::Message
* msg3
= GenerateReply();
232 // Start three requests. Since the proxy resolver is async, all the
233 // requests will be pending.
235 helper_
->OnResolveProxy(url1
, msg1
);
237 // Finish ProxyService's initialization.
238 ASSERT_EQ(1u, resolver_factory_
->pending_requests().size());
239 resolver_factory_
->pending_requests()[0]->CompleteNowWithForwarder(
240 net::OK
, &resolver_
);
242 helper_
->OnResolveProxy(url2
, msg2
);
243 helper_
->OnResolveProxy(url3
, msg3
);
245 // ResolveProxyHelper only keeps 1 request outstanding in ProxyService
247 ASSERT_EQ(1u, resolver_
.pending_requests().size());
248 EXPECT_EQ(url1
, resolver_
.pending_requests()[0]->url());
250 // Delete the underlying ResolveProxyMsgHelper -- this should cancel all
251 // the requests which are outstanding.
254 // The pending requests sent to the proxy resolver should have been cancelled.
256 EXPECT_EQ(0u, resolver_
.pending_requests().size());
258 EXPECT_TRUE(pending_result() == NULL
);
260 // It should also be the case that msg1, msg2, msg3 were deleted by the
261 // cancellation. (Else will show up as a leak in Valgrind).
264 } // namespace content