• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2013 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.
4 
5 #include "net/proxy/proxy_resolver_v8_tracing.h"
6 
7 #include "base/files/file_util.h"
8 #include "base/message_loop/message_loop.h"
9 #include "base/path_service.h"
10 #include "base/stl_util.h"
11 #include "base/strings/string_util.h"
12 #include "base/strings/stringprintf.h"
13 #include "base/strings/utf_string_conversions.h"
14 #include "base/synchronization/waitable_event.h"
15 #include "base/threading/platform_thread.h"
16 #include "base/values.h"
17 #include "net/base/net_errors.h"
18 #include "net/base/net_log.h"
19 #include "net/base/net_log_unittest.h"
20 #include "net/base/test_completion_callback.h"
21 #include "net/dns/host_cache.h"
22 #include "net/dns/mock_host_resolver.h"
23 #include "net/proxy/proxy_info.h"
24 #include "net/proxy/proxy_resolver_error_observer.h"
25 #include "testing/gtest/include/gtest/gtest.h"
26 #include "url/gurl.h"
27 
28 namespace net {
29 
30 namespace {
31 
32 class ProxyResolverV8TracingTest : public testing::Test {
33  public:
TearDown()34   virtual void TearDown() OVERRIDE {
35     // Drain any pending messages, which may be left over from cancellation.
36     // This way they get reliably run as part of the current test, rather than
37     // spilling into the next test's execution.
38     base::MessageLoop::current()->RunUntilIdle();
39   }
40 };
41 
LoadScriptData(const char * filename)42 scoped_refptr<ProxyResolverScriptData> LoadScriptData(const char* filename) {
43   base::FilePath path;
44   PathService::Get(base::DIR_SOURCE_ROOT, &path);
45   path = path.AppendASCII("net");
46   path = path.AppendASCII("data");
47   path = path.AppendASCII("proxy_resolver_v8_tracing_unittest");
48   path = path.AppendASCII(filename);
49 
50   // Try to read the file from disk.
51   std::string file_contents;
52   bool ok = base::ReadFileToString(path, &file_contents);
53 
54   // If we can't load the file from disk, something is misconfigured.
55   EXPECT_TRUE(ok) << "Failed to read file: " << path.value();
56 
57   // Load the PAC script into the ProxyResolver.
58   return ProxyResolverScriptData::FromUTF8(file_contents);
59 }
60 
InitResolver(ProxyResolverV8Tracing * resolver,const char * filename)61 void InitResolver(ProxyResolverV8Tracing* resolver, const char* filename) {
62   TestCompletionCallback callback;
63   int rv =
64       resolver->SetPacScript(LoadScriptData(filename), callback.callback());
65   EXPECT_EQ(ERR_IO_PENDING, rv);
66   EXPECT_EQ(OK, callback.WaitForResult());
67 }
68 
69 class MockErrorObserver : public ProxyResolverErrorObserver {
70  public:
MockErrorObserver()71   MockErrorObserver() : event_(true, false) {}
72 
OnPACScriptError(int line_number,const base::string16 & error)73   virtual void OnPACScriptError(int line_number,
74                                 const base::string16& error) OVERRIDE {
75     {
76       base::AutoLock l(lock_);
77       output += base::StringPrintf("Error: line %d: %s\n", line_number,
78                                    base::UTF16ToASCII(error).c_str());
79     }
80     event_.Signal();
81   }
82 
GetOutput()83   std::string GetOutput() {
84     base::AutoLock l(lock_);
85     return output;
86   }
87 
WaitForOutput()88   void WaitForOutput() {
89     event_.Wait();
90   }
91 
92  private:
93   base::Lock lock_;
94   std::string output;
95 
96   base::WaitableEvent event_;
97 };
98 
TEST_F(ProxyResolverV8TracingTest,Simple)99 TEST_F(ProxyResolverV8TracingTest, Simple) {
100   CapturingNetLog log;
101   CapturingBoundNetLog request_log;
102   MockCachingHostResolver host_resolver;
103   MockErrorObserver* error_observer = new MockErrorObserver;
104   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
105 
106   InitResolver(&resolver, "simple.js");
107 
108   TestCompletionCallback callback;
109   ProxyInfo proxy_info;
110 
111   int rv = resolver.GetProxyForURL(
112       GURL("http://foo/"), &proxy_info, callback.callback(),
113       NULL, request_log.bound());
114 
115   EXPECT_EQ(ERR_IO_PENDING, rv);
116   EXPECT_EQ(OK, callback.WaitForResult());
117 
118   EXPECT_EQ("foo:99", proxy_info.proxy_server().ToURI());
119 
120   EXPECT_EQ(0u, host_resolver.num_resolve());
121 
122   // There were no errors.
123   EXPECT_EQ("", error_observer->GetOutput());
124 
125   // Check the NetLogs -- nothing was logged.
126   EXPECT_EQ(0u, log.GetSize());
127   EXPECT_EQ(0u, request_log.GetSize());
128 }
129 
TEST_F(ProxyResolverV8TracingTest,JavascriptError)130 TEST_F(ProxyResolverV8TracingTest, JavascriptError) {
131   CapturingNetLog log;
132   CapturingBoundNetLog request_log;
133   MockCachingHostResolver host_resolver;
134   MockErrorObserver* error_observer = new MockErrorObserver;
135   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
136 
137   InitResolver(&resolver, "error.js");
138 
139   TestCompletionCallback callback;
140   ProxyInfo proxy_info;
141 
142   int rv = resolver.GetProxyForURL(
143       GURL("http://throw-an-error/"), &proxy_info, callback.callback(), NULL,
144       request_log.bound());
145 
146   EXPECT_EQ(ERR_IO_PENDING, rv);
147   EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, callback.WaitForResult());
148 
149   EXPECT_EQ(0u, host_resolver.num_resolve());
150 
151   EXPECT_EQ("Error: line 5: Uncaught TypeError: Cannot read property 'split' "
152             "of null\n", error_observer->GetOutput());
153 
154   // Check the NetLogs -- there was 1 alert and 1 javascript error, and they
155   // were output to both the global log, and per-request log.
156   CapturingNetLog::CapturedEntryList entries_list[2];
157   log.GetEntries(&entries_list[0]);
158   request_log.GetEntries(&entries_list[1]);
159 
160   for (size_t list_i = 0; list_i < arraysize(entries_list); list_i++) {
161     const CapturingNetLog::CapturedEntryList& entries = entries_list[list_i];
162     EXPECT_EQ(2u, entries.size());
163     EXPECT_TRUE(
164         LogContainsEvent(entries, 0, NetLog::TYPE_PAC_JAVASCRIPT_ALERT,
165                          NetLog::PHASE_NONE));
166     EXPECT_TRUE(
167         LogContainsEvent(entries, 1, NetLog::TYPE_PAC_JAVASCRIPT_ERROR,
168                          NetLog::PHASE_NONE));
169 
170     EXPECT_EQ("{\"message\":\"Prepare to DIE!\"}", entries[0].GetParamsJson());
171     EXPECT_EQ("{\"line_number\":5,\"message\":\"Uncaught TypeError: Cannot "
172               "read property 'split' of null\"}", entries[1].GetParamsJson());
173   }
174 }
175 
TEST_F(ProxyResolverV8TracingTest,TooManyAlerts)176 TEST_F(ProxyResolverV8TracingTest, TooManyAlerts) {
177   CapturingNetLog log;
178   CapturingBoundNetLog request_log;
179   MockCachingHostResolver host_resolver;
180   MockErrorObserver* error_observer = new MockErrorObserver;
181   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
182 
183   InitResolver(&resolver, "too_many_alerts.js");
184 
185   TestCompletionCallback callback;
186   ProxyInfo proxy_info;
187 
188   int rv = resolver.GetProxyForURL(
189       GURL("http://foo/"),
190       &proxy_info,
191       callback.callback(),
192       NULL,
193       request_log.bound());
194 
195   EXPECT_EQ(ERR_IO_PENDING, rv);
196   EXPECT_EQ(OK, callback.WaitForResult());
197 
198   // Iteration1 does a DNS resolve
199   // Iteration2 exceeds the alert buffer
200   // Iteration3 runs in blocking mode and completes
201   EXPECT_EQ("foo:3", proxy_info.proxy_server().ToURI());
202 
203   EXPECT_EQ(1u, host_resolver.num_resolve());
204 
205   // No errors.
206   EXPECT_EQ("", error_observer->GetOutput());
207 
208   // Check the NetLogs -- the script generated 50 alerts, which were mirrored
209   // to both the global and per-request logs.
210   CapturingNetLog::CapturedEntryList entries_list[2];
211   log.GetEntries(&entries_list[0]);
212   request_log.GetEntries(&entries_list[1]);
213 
214   for (size_t list_i = 0; list_i < arraysize(entries_list); list_i++) {
215     const CapturingNetLog::CapturedEntryList& entries = entries_list[list_i];
216     EXPECT_EQ(50u, entries.size());
217     for (size_t i = 0; i < entries.size(); ++i) {
218       ASSERT_TRUE(
219           LogContainsEvent(entries, i, NetLog::TYPE_PAC_JAVASCRIPT_ALERT,
220                            NetLog::PHASE_NONE));
221     }
222   }
223 }
224 
225 // Verify that buffered alerts cannot grow unboundedly, even when the message is
226 // empty string.
TEST_F(ProxyResolverV8TracingTest,TooManyEmptyAlerts)227 TEST_F(ProxyResolverV8TracingTest, TooManyEmptyAlerts) {
228   CapturingNetLog log;
229   CapturingBoundNetLog request_log;
230   MockCachingHostResolver host_resolver;
231   MockErrorObserver* error_observer = new MockErrorObserver;
232   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
233 
234   InitResolver(&resolver, "too_many_empty_alerts.js");
235 
236   TestCompletionCallback callback;
237   ProxyInfo proxy_info;
238 
239   int rv = resolver.GetProxyForURL(
240       GURL("http://foo/"),
241       &proxy_info,
242       callback.callback(),
243       NULL,
244       request_log.bound());
245 
246   EXPECT_EQ(ERR_IO_PENDING, rv);
247   EXPECT_EQ(OK, callback.WaitForResult());
248 
249   EXPECT_EQ("foo:3", proxy_info.proxy_server().ToURI());
250 
251   EXPECT_EQ(1u, host_resolver.num_resolve());
252 
253   // No errors.
254   EXPECT_EQ("", error_observer->GetOutput());
255 
256   // Check the NetLogs -- the script generated 50 alerts, which were mirrored
257   // to both the global and per-request logs.
258   CapturingNetLog::CapturedEntryList entries_list[2];
259   log.GetEntries(&entries_list[0]);
260   request_log.GetEntries(&entries_list[1]);
261 
262   for (size_t list_i = 0; list_i < arraysize(entries_list); list_i++) {
263     const CapturingNetLog::CapturedEntryList& entries = entries_list[list_i];
264     EXPECT_EQ(1000u, entries.size());
265     for (size_t i = 0; i < entries.size(); ++i) {
266       ASSERT_TRUE(
267           LogContainsEvent(entries, i, NetLog::TYPE_PAC_JAVASCRIPT_ALERT,
268                            NetLog::PHASE_NONE));
269     }
270   }
271 }
272 
273 // This test runs a PAC script that issues a sequence of DNS resolves. The test
274 // verifies the final result, and that the underlying DNS resolver received
275 // the correct set of queries.
TEST_F(ProxyResolverV8TracingTest,Dns)276 TEST_F(ProxyResolverV8TracingTest, Dns) {
277   CapturingNetLog log;
278   CapturingBoundNetLog request_log;
279   MockCachingHostResolver host_resolver;
280   MockErrorObserver* error_observer = new MockErrorObserver;
281   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
282 
283   host_resolver.rules()->AddRuleForAddressFamily(
284       "host1", ADDRESS_FAMILY_IPV4, "166.155.144.44");
285   host_resolver.rules()
286       ->AddIPLiteralRule("host1", "::1,192.168.1.1", std::string());
287   host_resolver.rules()->AddSimulatedFailure("host2");
288   host_resolver.rules()->AddRule("host3", "166.155.144.33");
289   host_resolver.rules()->AddRule("host5", "166.155.144.55");
290   host_resolver.rules()->AddSimulatedFailure("host6");
291   host_resolver.rules()->AddRuleForAddressFamily(
292       "*", ADDRESS_FAMILY_IPV4, "122.133.144.155");
293   host_resolver.rules()->AddRule("*", "133.122.100.200");
294 
295   InitResolver(&resolver, "dns.js");
296 
297   TestCompletionCallback callback;
298   ProxyInfo proxy_info;
299 
300   int rv = resolver.GetProxyForURL(
301       GURL("http://foo/"),
302       &proxy_info,
303       callback.callback(),
304       NULL,
305       request_log.bound());
306 
307   EXPECT_EQ(ERR_IO_PENDING, rv);
308   EXPECT_EQ(OK, callback.WaitForResult());
309 
310   // The test does 13 DNS resolution, however only 7 of them are unique.
311   EXPECT_EQ(7u, host_resolver.num_resolve());
312 
313   const char* kExpectedResult =
314     "122.133.144.155-"  // myIpAddress()
315     "null-"  // dnsResolve('')
316     "__1_192.168.1.1-"  // dnsResolveEx('host1')
317     "null-"  // dnsResolve('host2')
318     "166.155.144.33-"  // dnsResolve('host3')
319     "122.133.144.155-"  // myIpAddress()
320     "166.155.144.33-"  // dnsResolve('host3')
321     "__1_192.168.1.1-"  // dnsResolveEx('host1')
322     "122.133.144.155-"  // myIpAddress()
323     "null-"  // dnsResolve('host2')
324     "-"  // dnsResolveEx('host6')
325     "133.122.100.200-"  // myIpAddressEx()
326     "166.155.144.44"  // dnsResolve('host1')
327     ":99";
328 
329   EXPECT_EQ(kExpectedResult, proxy_info.proxy_server().ToURI());
330 
331   // No errors.
332   EXPECT_EQ("", error_observer->GetOutput());
333 
334   // Check the NetLogs -- the script generated 1 alert, mirrored to both
335   // the per-request and global logs.
336   CapturingNetLog::CapturedEntryList entries_list[2];
337   log.GetEntries(&entries_list[0]);
338   request_log.GetEntries(&entries_list[1]);
339 
340   for (size_t list_i = 0; list_i < arraysize(entries_list); list_i++) {
341     const CapturingNetLog::CapturedEntryList& entries = entries_list[list_i];
342     EXPECT_EQ(1u, entries.size());
343     EXPECT_TRUE(
344         LogContainsEvent(entries, 0, NetLog::TYPE_PAC_JAVASCRIPT_ALERT,
345                          NetLog::PHASE_NONE));
346     EXPECT_EQ("{\"message\":\"iteration: 7\"}", entries[0].GetParamsJson());
347   }
348 }
349 
350 // This test runs a PAC script that does "myIpAddress()" followed by
351 // "dnsResolve()". This requires 2 restarts. However once the HostResolver's
352 // cache is warmed, subsequent calls should take 0 restarts.
TEST_F(ProxyResolverV8TracingTest,DnsChecksCache)353 TEST_F(ProxyResolverV8TracingTest, DnsChecksCache) {
354   CapturingNetLog log;
355   CapturingBoundNetLog request_log;
356   MockCachingHostResolver host_resolver;
357   MockErrorObserver* error_observer = new MockErrorObserver;
358   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
359 
360   host_resolver.rules()->AddRule("foopy", "166.155.144.11");
361   host_resolver.rules()->AddRule("*", "122.133.144.155");
362 
363   InitResolver(&resolver, "simple_dns.js");
364 
365   TestCompletionCallback callback1;
366   TestCompletionCallback callback2;
367   ProxyInfo proxy_info;
368 
369   int rv = resolver.GetProxyForURL(
370       GURL("http://foopy/req1"),
371       &proxy_info,
372       callback1.callback(),
373       NULL,
374       request_log.bound());
375 
376   EXPECT_EQ(ERR_IO_PENDING, rv);
377   EXPECT_EQ(OK, callback1.WaitForResult());
378 
379   // The test does 2 DNS resolutions.
380   EXPECT_EQ(2u, host_resolver.num_resolve());
381 
382   // The first request took 2 restarts, hence on g_iteration=3.
383   EXPECT_EQ("166.155.144.11:3", proxy_info.proxy_server().ToURI());
384 
385   rv = resolver.GetProxyForURL(
386       GURL("http://foopy/req2"),
387       &proxy_info,
388       callback2.callback(),
389       NULL,
390       request_log.bound());
391 
392   EXPECT_EQ(ERR_IO_PENDING, rv);
393   EXPECT_EQ(OK, callback2.WaitForResult());
394 
395   EXPECT_EQ(4u, host_resolver.num_resolve());
396 
397   // This time no restarts were required, so g_iteration incremented by 1.
398   EXPECT_EQ("166.155.144.11:4", proxy_info.proxy_server().ToURI());
399 
400   // No errors.
401   EXPECT_EQ("", error_observer->GetOutput());
402 
403   EXPECT_EQ(0u, log.GetSize());
404   EXPECT_EQ(0u, request_log.GetSize());
405 }
406 
407 // This test runs a weird PAC script that was designed to defeat the DNS tracing
408 // optimization. The proxy resolver should detect the inconsistency and
409 // fall-back to synchronous mode execution.
TEST_F(ProxyResolverV8TracingTest,FallBackToSynchronous1)410 TEST_F(ProxyResolverV8TracingTest, FallBackToSynchronous1) {
411   CapturingNetLog log;
412   CapturingBoundNetLog request_log;
413   MockCachingHostResolver host_resolver;
414   MockErrorObserver* error_observer = new MockErrorObserver;
415   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
416 
417   host_resolver.rules()->AddRule("host1", "166.155.144.11");
418   host_resolver.rules()->AddRule("crazy4", "133.199.111.4");
419   host_resolver.rules()->AddRule("*", "122.133.144.155");
420 
421   InitResolver(&resolver, "global_sideffects1.js");
422 
423   TestCompletionCallback callback;
424   ProxyInfo proxy_info;
425 
426   int rv = resolver.GetProxyForURL(
427       GURL("http://foo/"), &proxy_info, callback.callback(), NULL,
428       request_log.bound());
429   EXPECT_EQ(ERR_IO_PENDING, rv);
430   EXPECT_EQ(OK, callback.WaitForResult());
431 
432   // The script itself only does 2 DNS resolves per execution, however it
433   // constructs the hostname using a global counter which changes on each
434   // invocation.
435   EXPECT_EQ(3u, host_resolver.num_resolve());
436 
437   EXPECT_EQ("166.155.144.11-133.199.111.4:100",
438             proxy_info.proxy_server().ToURI());
439 
440   // No errors.
441   EXPECT_EQ("", error_observer->GetOutput());
442 
443   // Check the NetLogs -- the script generated 1 alert, mirrored to both
444   // the per-request and global logs.
445   CapturingNetLog::CapturedEntryList entries_list[2];
446   log.GetEntries(&entries_list[0]);
447   request_log.GetEntries(&entries_list[1]);
448 
449   for (size_t list_i = 0; list_i < arraysize(entries_list); list_i++) {
450     const CapturingNetLog::CapturedEntryList& entries = entries_list[list_i];
451     EXPECT_EQ(1u, entries.size());
452     EXPECT_TRUE(
453         LogContainsEvent(entries, 0, NetLog::TYPE_PAC_JAVASCRIPT_ALERT,
454                          NetLog::PHASE_NONE));
455     EXPECT_EQ("{\"message\":\"iteration: 4\"}", entries[0].GetParamsJson());
456   }
457 }
458 
459 // This test runs a weird PAC script that was designed to defeat the DNS tracing
460 // optimization. The proxy resolver should detect the inconsistency and
461 // fall-back to synchronous mode execution.
TEST_F(ProxyResolverV8TracingTest,FallBackToSynchronous2)462 TEST_F(ProxyResolverV8TracingTest, FallBackToSynchronous2) {
463   CapturingNetLog log;
464   CapturingBoundNetLog request_log;
465   MockCachingHostResolver host_resolver;
466   MockErrorObserver* error_observer = new MockErrorObserver;
467   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
468 
469   host_resolver.rules()->AddRule("host1", "166.155.144.11");
470   host_resolver.rules()->AddRule("host2", "166.155.144.22");
471   host_resolver.rules()->AddRule("host3", "166.155.144.33");
472   host_resolver.rules()->AddRule("host4", "166.155.144.44");
473   host_resolver.rules()->AddRule("*", "122.133.144.155");
474 
475   InitResolver(&resolver, "global_sideffects2.js");
476 
477   TestCompletionCallback callback;
478   ProxyInfo proxy_info;
479 
480   int rv = resolver.GetProxyForURL(
481       GURL("http://foo/"), &proxy_info, callback.callback(), NULL,
482       request_log.bound());
483   EXPECT_EQ(ERR_IO_PENDING, rv);
484   EXPECT_EQ(OK, callback.WaitForResult());
485 
486   EXPECT_EQ(3u, host_resolver.num_resolve());
487 
488   EXPECT_EQ("166.155.144.44:100", proxy_info.proxy_server().ToURI());
489 
490   // No errors.
491   EXPECT_EQ("", error_observer->GetOutput());
492 
493   // Check the NetLogs -- nothing was logged.
494   EXPECT_EQ(0u, log.GetSize());
495   EXPECT_EQ(0u, request_log.GetSize());
496 }
497 
498 // This test runs a weird PAC script that yields a never ending sequence
499 // of DNS resolves when restarting. Running it will hit the maximum
500 // DNS resolves per request limit (20) after which every DNS resolve will
501 // fail.
TEST_F(ProxyResolverV8TracingTest,InfiniteDNSSequence)502 TEST_F(ProxyResolverV8TracingTest, InfiniteDNSSequence) {
503   CapturingNetLog log;
504   CapturingBoundNetLog request_log;
505   MockCachingHostResolver host_resolver;
506   MockErrorObserver* error_observer = new MockErrorObserver;
507   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
508 
509   host_resolver.rules()->AddRule("host*", "166.155.144.11");
510   host_resolver.rules()->AddRule("*", "122.133.144.155");
511 
512   InitResolver(&resolver, "global_sideffects3.js");
513 
514   TestCompletionCallback callback;
515   ProxyInfo proxy_info;
516 
517   int rv = resolver.GetProxyForURL(
518       GURL("http://foo/"), &proxy_info, callback.callback(), NULL,
519       request_log.bound());
520   EXPECT_EQ(ERR_IO_PENDING, rv);
521   EXPECT_EQ(OK, callback.WaitForResult());
522 
523   EXPECT_EQ(20u, host_resolver.num_resolve());
524 
525   EXPECT_EQ(
526       "166.155.144.11-166.155.144.11-166.155.144.11-166.155.144.11-"
527       "166.155.144.11-166.155.144.11-166.155.144.11-166.155.144.11-"
528       "166.155.144.11-166.155.144.11-166.155.144.11-166.155.144.11-"
529       "166.155.144.11-166.155.144.11-166.155.144.11-166.155.144.11-"
530       "166.155.144.11-166.155.144.11-166.155.144.11-166.155.144.11-"
531       "null:21", proxy_info.proxy_server().ToURI());
532 
533   // No errors.
534   EXPECT_EQ("", error_observer->GetOutput());
535 
536   // Check the NetLogs -- 1 alert was logged.
537   EXPECT_EQ(1u, log.GetSize());
538   EXPECT_EQ(1u, request_log.GetSize());
539 }
540 
541 // This test runs a weird PAC script that yields a never ending sequence
542 // of DNS resolves when restarting. Running it will hit the maximum
543 // DNS resolves per request limit (20) after which every DNS resolve will
544 // fail.
TEST_F(ProxyResolverV8TracingTest,InfiniteDNSSequence2)545 TEST_F(ProxyResolverV8TracingTest, InfiniteDNSSequence2) {
546   CapturingNetLog log;
547   CapturingBoundNetLog request_log;
548   MockCachingHostResolver host_resolver;
549   MockErrorObserver* error_observer = new MockErrorObserver;
550   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
551 
552   host_resolver.rules()->AddRule("host*", "166.155.144.11");
553   host_resolver.rules()->AddRule("*", "122.133.144.155");
554 
555   InitResolver(&resolver, "global_sideffects4.js");
556 
557   TestCompletionCallback callback;
558   ProxyInfo proxy_info;
559 
560   int rv = resolver.GetProxyForURL(
561       GURL("http://foo/"), &proxy_info, callback.callback(), NULL,
562       request_log.bound());
563   EXPECT_EQ(ERR_IO_PENDING, rv);
564   EXPECT_EQ(OK, callback.WaitForResult());
565 
566   EXPECT_EQ(20u, host_resolver.num_resolve());
567 
568   EXPECT_EQ("null21:34", proxy_info.proxy_server().ToURI());
569 
570   // No errors.
571   EXPECT_EQ("", error_observer->GetOutput());
572 
573   // Check the NetLogs -- 1 alert was logged.
574   EXPECT_EQ(1u, log.GetSize());
575   EXPECT_EQ(1u, request_log.GetSize());
576 }
577 
DnsDuringInitHelper(bool synchronous_host_resolver)578 void DnsDuringInitHelper(bool synchronous_host_resolver) {
579   CapturingNetLog log;
580   CapturingBoundNetLog request_log;
581   MockCachingHostResolver host_resolver;
582   host_resolver.set_synchronous_mode(synchronous_host_resolver);
583   MockErrorObserver* error_observer = new MockErrorObserver;
584   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
585 
586   host_resolver.rules()->AddRule("host1", "91.13.12.1");
587   host_resolver.rules()->AddRule("host2", "91.13.12.2");
588 
589   InitResolver(&resolver, "dns_during_init.js");
590 
591   // Initialization did 2 dnsResolves.
592   EXPECT_EQ(2u, host_resolver.num_resolve());
593 
594   host_resolver.rules()->ClearRules();
595   host_resolver.GetHostCache()->clear();
596 
597   host_resolver.rules()->AddRule("host1", "145.88.13.3");
598   host_resolver.rules()->AddRule("host2", "137.89.8.45");
599 
600   TestCompletionCallback callback;
601   ProxyInfo proxy_info;
602 
603   int rv = resolver.GetProxyForURL(
604       GURL("http://foo/"), &proxy_info, callback.callback(), NULL,
605       request_log.bound());
606   EXPECT_EQ(ERR_IO_PENDING, rv);
607   EXPECT_EQ(OK, callback.WaitForResult());
608 
609   // Fetched host1 and host2 again, since the ones done during initialization
610   // should not have been cached.
611   EXPECT_EQ(4u, host_resolver.num_resolve());
612 
613   EXPECT_EQ("91.13.12.1-91.13.12.2-145.88.13.3-137.89.8.45:99",
614             proxy_info.proxy_server().ToURI());
615 
616   // Check the NetLogs -- the script generated 2 alerts during initialization.
617   EXPECT_EQ(0u, request_log.GetSize());
618   CapturingNetLog::CapturedEntryList entries;
619   log.GetEntries(&entries);
620 
621   ASSERT_EQ(2u, entries.size());
622   EXPECT_TRUE(
623       LogContainsEvent(entries, 0, NetLog::TYPE_PAC_JAVASCRIPT_ALERT,
624                        NetLog::PHASE_NONE));
625   EXPECT_TRUE(
626       LogContainsEvent(entries, 1, NetLog::TYPE_PAC_JAVASCRIPT_ALERT,
627                        NetLog::PHASE_NONE));
628 
629   EXPECT_EQ("{\"message\":\"Watsup\"}", entries[0].GetParamsJson());
630   EXPECT_EQ("{\"message\":\"Watsup2\"}", entries[1].GetParamsJson());
631 }
632 
633 // Tests a PAC script which does DNS resolves during initialization.
TEST_F(ProxyResolverV8TracingTest,DnsDuringInit)634 TEST_F(ProxyResolverV8TracingTest, DnsDuringInit) {
635   // Test with both both a host resolver that always completes asynchronously,
636   // and then again with one that completes synchronously.
637   DnsDuringInitHelper(false);
638   DnsDuringInitHelper(true);
639 }
640 
CrashCallback(int)641 void CrashCallback(int) {
642   // Be extra sure that if the callback ever gets invoked, the test will fail.
643   CHECK(false);
644 }
645 
646 // Start some requests, cancel them all, and then destroy the resolver.
647 // Note the execution order for this test can vary. Since multiple
648 // threads are involved, the cancellation may be received a different
649 // times.
TEST_F(ProxyResolverV8TracingTest,CancelAll)650 TEST_F(ProxyResolverV8TracingTest, CancelAll) {
651   MockCachingHostResolver host_resolver;
652   MockErrorObserver* error_observer = new MockErrorObserver;
653   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, NULL);
654 
655   host_resolver.rules()->AddSimulatedFailure("*");
656 
657   InitResolver(&resolver, "dns.js");
658 
659   const size_t kNumRequests = 5;
660   ProxyInfo proxy_info[kNumRequests];
661   ProxyResolver::RequestHandle request[kNumRequests];
662 
663   for (size_t i = 0; i < kNumRequests; ++i) {
664     int rv = resolver.GetProxyForURL(
665         GURL("http://foo/"), &proxy_info[i],
666         base::Bind(&CrashCallback), &request[i], BoundNetLog());
667     EXPECT_EQ(ERR_IO_PENDING, rv);
668   }
669 
670   for (size_t i = 0; i < kNumRequests; ++i) {
671     resolver.CancelRequest(request[i]);
672   }
673 }
674 
675 // Note the execution order for this test can vary. Since multiple
676 // threads are involved, the cancellation may be received a different
677 // times.
TEST_F(ProxyResolverV8TracingTest,CancelSome)678 TEST_F(ProxyResolverV8TracingTest, CancelSome) {
679   MockCachingHostResolver host_resolver;
680   MockErrorObserver* error_observer = new MockErrorObserver;
681   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, NULL);
682 
683   host_resolver.rules()->AddSimulatedFailure("*");
684 
685   InitResolver(&resolver, "dns.js");
686 
687   ProxyInfo proxy_info1;
688   ProxyInfo proxy_info2;
689   ProxyResolver::RequestHandle request1;
690   ProxyResolver::RequestHandle request2;
691   TestCompletionCallback callback;
692 
693   int rv = resolver.GetProxyForURL(
694       GURL("http://foo/"), &proxy_info1,
695       base::Bind(&CrashCallback), &request1, BoundNetLog());
696   EXPECT_EQ(ERR_IO_PENDING, rv);
697 
698   rv = resolver.GetProxyForURL(
699       GURL("http://foo/"), &proxy_info2,
700       callback.callback(), &request2, BoundNetLog());
701   EXPECT_EQ(ERR_IO_PENDING, rv);
702 
703   resolver.CancelRequest(request1);
704 
705   EXPECT_EQ(OK, callback.WaitForResult());
706 }
707 
708 // Cancel a request after it has finished running on the worker thread, and has
709 // posted a task the completion task back to origin thread.
TEST_F(ProxyResolverV8TracingTest,CancelWhilePendingCompletionTask)710 TEST_F(ProxyResolverV8TracingTest, CancelWhilePendingCompletionTask) {
711   MockCachingHostResolver host_resolver;
712   MockErrorObserver* error_observer = new MockErrorObserver;
713   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, NULL);
714 
715   host_resolver.rules()->AddSimulatedFailure("*");
716 
717   InitResolver(&resolver, "error.js");
718 
719   ProxyInfo proxy_info1;
720   ProxyInfo proxy_info2;
721   ProxyInfo proxy_info3;
722   ProxyResolver::RequestHandle request1;
723   ProxyResolver::RequestHandle request2;
724   ProxyResolver::RequestHandle request3;
725   TestCompletionCallback callback;
726 
727   int rv = resolver.GetProxyForURL(
728       GURL("http://foo/"), &proxy_info1,
729       base::Bind(&CrashCallback), &request1, BoundNetLog());
730   EXPECT_EQ(ERR_IO_PENDING, rv);
731 
732   rv = resolver.GetProxyForURL(
733       GURL("http://throw-an-error/"), &proxy_info2,
734       callback.callback(), &request2, BoundNetLog());
735   EXPECT_EQ(ERR_IO_PENDING, rv);
736 
737   // Wait until the first request has finished running on the worker thread.
738   // (The second request will output an error).
739   error_observer->WaitForOutput();
740 
741   // Cancel the first request, while it has a pending completion task on
742   // the origin thread.
743   resolver.CancelRequest(request1);
744 
745   EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, callback.WaitForResult());
746 
747   // Start another request, to make sure it is able to complete.
748   rv = resolver.GetProxyForURL(
749       GURL("http://i-have-no-idea-what-im-doing/"), &proxy_info3,
750       callback.callback(), &request3, BoundNetLog());
751   EXPECT_EQ(ERR_IO_PENDING, rv);
752 
753   EXPECT_EQ(OK, callback.WaitForResult());
754 
755   EXPECT_EQ("i-approve-this-message:42",
756             proxy_info3.proxy_server().ToURI());
757 }
758 
759 // This implementation of HostResolver allows blocking until a resolve request
760 // has been received. The resolve requests it receives will never be completed.
761 class BlockableHostResolver : public HostResolver {
762  public:
BlockableHostResolver()763   BlockableHostResolver()
764       : num_cancelled_requests_(0), waiting_for_resolve_(false) {}
765 
Resolve(const RequestInfo & info,RequestPriority priority,AddressList * addresses,const CompletionCallback & callback,RequestHandle * out_req,const BoundNetLog & net_log)766   virtual int Resolve(const RequestInfo& info,
767                       RequestPriority priority,
768                       AddressList* addresses,
769                       const CompletionCallback& callback,
770                       RequestHandle* out_req,
771                       const BoundNetLog& net_log) OVERRIDE {
772     EXPECT_FALSE(callback.is_null());
773     EXPECT_TRUE(out_req);
774 
775     if (!action_.is_null())
776       action_.Run();
777 
778     // Indicate to the caller that a request was received.
779     EXPECT_TRUE(waiting_for_resolve_);
780     base::MessageLoop::current()->Quit();
781 
782     // This line is intentionally after action_.Run(), since one of the
783     // tests does a cancellation inside of Resolve(), and it is more
784     // interesting if *out_req hasn't been written yet at that point.
785     *out_req = reinterpret_cast<RequestHandle*>(1);  // Magic value.
786 
787     // Return ERR_IO_PENDING as this request will NEVER be completed.
788     // Expectation is for the caller to later cancel the request.
789     return ERR_IO_PENDING;
790   }
791 
ResolveFromCache(const RequestInfo & info,AddressList * addresses,const BoundNetLog & net_log)792   virtual int ResolveFromCache(const RequestInfo& info,
793                                AddressList* addresses,
794                                const BoundNetLog& net_log) OVERRIDE {
795     NOTREACHED();
796     return ERR_DNS_CACHE_MISS;
797   }
798 
CancelRequest(RequestHandle req)799   virtual void CancelRequest(RequestHandle req) OVERRIDE {
800     EXPECT_EQ(reinterpret_cast<RequestHandle*>(1), req);
801     num_cancelled_requests_++;
802   }
803 
SetAction(const base::Callback<void (void)> & action)804   void SetAction(const base::Callback<void(void)>& action) {
805     action_ = action;
806   }
807 
808   // Waits until Resolve() has been called.
WaitUntilRequestIsReceived()809   void WaitUntilRequestIsReceived() {
810     waiting_for_resolve_ = true;
811     base::MessageLoop::current()->Run();
812     DCHECK(waiting_for_resolve_);
813     waiting_for_resolve_ = false;
814   }
815 
num_cancelled_requests() const816   int num_cancelled_requests() const {
817     return num_cancelled_requests_;
818   }
819 
820  private:
821   int num_cancelled_requests_;
822   bool waiting_for_resolve_;
823   base::Callback<void(void)> action_;
824 };
825 
826 // This cancellation test exercises a more predictable cancellation codepath --
827 // when the request has an outstanding DNS request in flight.
TEST_F(ProxyResolverV8TracingTest,CancelWhileOutstandingNonBlockingDns)828 TEST_F(ProxyResolverV8TracingTest, CancelWhileOutstandingNonBlockingDns) {
829   BlockableHostResolver host_resolver;
830   MockErrorObserver* error_observer = new MockErrorObserver;
831   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, NULL);
832 
833   InitResolver(&resolver, "dns.js");
834 
835   ProxyInfo proxy_info1;
836   ProxyInfo proxy_info2;
837   ProxyResolver::RequestHandle request1;
838   ProxyResolver::RequestHandle request2;
839 
840   int rv = resolver.GetProxyForURL(
841       GURL("http://foo/req1"), &proxy_info1,
842       base::Bind(&CrashCallback), &request1, BoundNetLog());
843 
844   EXPECT_EQ(ERR_IO_PENDING, rv);
845 
846   host_resolver.WaitUntilRequestIsReceived();
847 
848   rv = resolver.GetProxyForURL(
849       GURL("http://foo/req2"), &proxy_info2,
850       base::Bind(&CrashCallback), &request2, BoundNetLog());
851 
852   EXPECT_EQ(ERR_IO_PENDING, rv);
853 
854   host_resolver.WaitUntilRequestIsReceived();
855 
856   resolver.CancelRequest(request1);
857   resolver.CancelRequest(request2);
858 
859   EXPECT_EQ(2, host_resolver.num_cancelled_requests());
860 
861   // After leaving this scope, the ProxyResolver is destroyed.
862   // This should not cause any problems, as the outstanding work
863   // should have been cancelled.
864 }
865 
CancelRequestAndPause(ProxyResolverV8Tracing * resolver,ProxyResolver::RequestHandle request)866 void CancelRequestAndPause(ProxyResolverV8Tracing* resolver,
867                            ProxyResolver::RequestHandle request) {
868   resolver->CancelRequest(request);
869 
870   // Sleep for a little bit. This makes it more likely for the worker
871   // thread to have returned from its call, and serves as a regression
872   // test for http://crbug.com/173373.
873   base::PlatformThread::Sleep(base::TimeDelta::FromMilliseconds(30));
874 }
875 
876 // In non-blocking mode, the worker thread actually does block for
877 // a short time to see if the result is in the DNS cache. Test
878 // cancellation while the worker thread is waiting on this event.
TEST_F(ProxyResolverV8TracingTest,CancelWhileBlockedInNonBlockingDns)879 TEST_F(ProxyResolverV8TracingTest, CancelWhileBlockedInNonBlockingDns) {
880   BlockableHostResolver host_resolver;
881   MockErrorObserver* error_observer = new MockErrorObserver;
882   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, NULL);
883 
884   InitResolver(&resolver, "dns.js");
885 
886   ProxyInfo proxy_info;
887   ProxyResolver::RequestHandle request;
888 
889   int rv = resolver.GetProxyForURL(
890       GURL("http://foo/"), &proxy_info,
891       base::Bind(&CrashCallback), &request, BoundNetLog());
892 
893   EXPECT_EQ(ERR_IO_PENDING, rv);
894 
895   host_resolver.SetAction(
896       base::Bind(CancelRequestAndPause, &resolver, request));
897 
898   host_resolver.WaitUntilRequestIsReceived();
899 
900   // At this point the host resolver ran Resolve(), and should have cancelled
901   // the request.
902 
903   EXPECT_EQ(1, host_resolver.num_cancelled_requests());
904 }
905 
906 // Cancel the request while there is a pending DNS request, however before
907 // the request is sent to the host resolver.
TEST_F(ProxyResolverV8TracingTest,CancelWhileBlockedInNonBlockingDns2)908 TEST_F(ProxyResolverV8TracingTest, CancelWhileBlockedInNonBlockingDns2) {
909   MockCachingHostResolver host_resolver;
910   MockErrorObserver* error_observer = new MockErrorObserver;
911   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, NULL);
912 
913   InitResolver(&resolver, "dns.js");
914 
915   ProxyInfo proxy_info;
916   ProxyResolver::RequestHandle request;
917 
918   int rv = resolver.GetProxyForURL(
919       GURL("http://foo/"), &proxy_info,
920       base::Bind(&CrashCallback), &request, BoundNetLog());
921 
922   EXPECT_EQ(ERR_IO_PENDING, rv);
923 
924   // Wait a bit, so the DNS task has hopefully been posted. The test will
925   // work whatever the delay is here, but it is most useful if the delay
926   // is large enough to allow a task to be posted back.
927   base::PlatformThread::Sleep(base::TimeDelta::FromMilliseconds(10));
928   resolver.CancelRequest(request);
929 
930   EXPECT_EQ(0u, host_resolver.num_resolve());
931 }
932 
TEST_F(ProxyResolverV8TracingTest,CancelSetPacWhileOutstandingBlockingDns)933 TEST_F(ProxyResolverV8TracingTest, CancelSetPacWhileOutstandingBlockingDns) {
934   BlockableHostResolver host_resolver;
935   MockErrorObserver* error_observer = new MockErrorObserver;
936 
937   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, NULL);
938 
939   int rv =
940       resolver.SetPacScript(LoadScriptData("dns_during_init.js"),
941       base::Bind(&CrashCallback));
942   EXPECT_EQ(ERR_IO_PENDING, rv);
943 
944   host_resolver.WaitUntilRequestIsReceived();
945 
946   resolver.CancelSetPacScript();
947   EXPECT_EQ(1, host_resolver.num_cancelled_requests());
948 }
949 
950 // This tests that the execution of a PAC script is terminated when the DNS
951 // dependencies are missing. If the test fails, then it will hang.
TEST_F(ProxyResolverV8TracingTest,Terminate)952 TEST_F(ProxyResolverV8TracingTest, Terminate) {
953   CapturingNetLog log;
954   CapturingBoundNetLog request_log;
955   MockCachingHostResolver host_resolver;
956   MockErrorObserver* error_observer = new MockErrorObserver;
957   ProxyResolverV8Tracing resolver(&host_resolver, error_observer, &log);
958 
959   host_resolver.rules()->AddRule("host1", "182.111.0.222");
960   host_resolver.rules()->AddRule("host2", "111.33.44.55");
961 
962   InitResolver(&resolver, "terminate.js");
963 
964   TestCompletionCallback callback;
965   ProxyInfo proxy_info;
966 
967   int rv = resolver.GetProxyForURL(
968       GURL("http://foopy/req1"),
969       &proxy_info,
970       callback.callback(),
971       NULL,
972       request_log.bound());
973 
974   EXPECT_EQ(ERR_IO_PENDING, rv);
975   EXPECT_EQ(OK, callback.WaitForResult());
976 
977   // The test does 2 DNS resolutions.
978   EXPECT_EQ(2u, host_resolver.num_resolve());
979 
980   EXPECT_EQ("foopy:3", proxy_info.proxy_server().ToURI());
981 
982   // No errors.
983   EXPECT_EQ("", error_observer->GetOutput());
984 
985   EXPECT_EQ(0u, log.GetSize());
986   EXPECT_EQ(0u, request_log.GetSize());
987 }
988 
989 // Tests that multiple instances of ProxyResolverV8Tracing can coexist and run
990 // correctly at the same time. This is relevant because at the moment (time
991 // this test was written) each ProxyResolverV8Tracing creates its own thread to
992 // run V8 on, however each thread is operating on the same v8::Isolate.
TEST_F(ProxyResolverV8TracingTest,MultipleResolvers)993 TEST_F(ProxyResolverV8TracingTest, MultipleResolvers) {
994   // ------------------------
995   // Setup resolver0
996   // ------------------------
997   MockHostResolver host_resolver0;
998   host_resolver0.rules()->AddRuleForAddressFamily(
999       "host1", ADDRESS_FAMILY_IPV4, "166.155.144.44");
1000   host_resolver0.rules()
1001       ->AddIPLiteralRule("host1", "::1,192.168.1.1", std::string());
1002   host_resolver0.rules()->AddSimulatedFailure("host2");
1003   host_resolver0.rules()->AddRule("host3", "166.155.144.33");
1004   host_resolver0.rules()->AddRule("host5", "166.155.144.55");
1005   host_resolver0.rules()->AddSimulatedFailure("host6");
1006   host_resolver0.rules()->AddRuleForAddressFamily(
1007       "*", ADDRESS_FAMILY_IPV4, "122.133.144.155");
1008   host_resolver0.rules()->AddRule("*", "133.122.100.200");
1009   ProxyResolverV8Tracing resolver0(
1010       &host_resolver0, new MockErrorObserver, NULL);
1011   InitResolver(&resolver0, "dns.js");
1012 
1013   // ------------------------
1014   // Setup resolver1
1015   // ------------------------
1016   ProxyResolverV8Tracing resolver1(
1017       &host_resolver0, new MockErrorObserver, NULL);
1018   InitResolver(&resolver1, "dns.js");
1019 
1020   // ------------------------
1021   // Setup resolver2
1022   // ------------------------
1023   ProxyResolverV8Tracing resolver2(
1024       &host_resolver0, new MockErrorObserver, NULL);
1025   InitResolver(&resolver2, "simple.js");
1026 
1027   // ------------------------
1028   // Setup resolver3
1029   // ------------------------
1030   MockHostResolver host_resolver3;
1031   host_resolver3.rules()->AddRule("foo", "166.155.144.33");
1032   ProxyResolverV8Tracing resolver3(
1033       &host_resolver3, new MockErrorObserver, NULL);
1034   InitResolver(&resolver3, "simple_dns.js");
1035 
1036   // ------------------------
1037   // Queue up work for each resolver (which will be running in parallel).
1038   // ------------------------
1039 
1040   ProxyResolverV8Tracing* resolver[] = {
1041     &resolver0, &resolver1, &resolver2, &resolver3,
1042   };
1043 
1044   const size_t kNumResolvers = arraysize(resolver);
1045   const size_t kNumIterations = 20;
1046   const size_t kNumResults = kNumResolvers * kNumIterations;
1047   TestCompletionCallback callback[kNumResults];
1048   ProxyInfo proxy_info[kNumResults];
1049 
1050   for (size_t i = 0; i < kNumResults; ++i) {
1051     size_t resolver_i = i % kNumResolvers;
1052     int rv = resolver[resolver_i]->GetProxyForURL(
1053         GURL("http://foo/"), &proxy_info[i], callback[i].callback(), NULL,
1054         BoundNetLog());
1055     EXPECT_EQ(ERR_IO_PENDING, rv);
1056   }
1057 
1058   // ------------------------
1059   // Verify all of the results.
1060   // ------------------------
1061 
1062   const char* kExpectedForDnsJs =
1063     "122.133.144.155-"  // myIpAddress()
1064     "null-"  // dnsResolve('')
1065     "__1_192.168.1.1-"  // dnsResolveEx('host1')
1066     "null-"  // dnsResolve('host2')
1067     "166.155.144.33-"  // dnsResolve('host3')
1068     "122.133.144.155-"  // myIpAddress()
1069     "166.155.144.33-"  // dnsResolve('host3')
1070     "__1_192.168.1.1-"  // dnsResolveEx('host1')
1071     "122.133.144.155-"  // myIpAddress()
1072     "null-"  // dnsResolve('host2')
1073     "-"  // dnsResolveEx('host6')
1074     "133.122.100.200-"  // myIpAddressEx()
1075     "166.155.144.44"  // dnsResolve('host1')
1076     ":99";
1077 
1078   for (size_t i = 0; i < kNumResults; ++i) {
1079     size_t resolver_i = i % kNumResolvers;
1080     EXPECT_EQ(OK, callback[i].WaitForResult());
1081 
1082     std::string proxy_uri = proxy_info[i].proxy_server().ToURI();
1083 
1084     if (resolver_i == 0 || resolver_i == 1) {
1085       EXPECT_EQ(kExpectedForDnsJs, proxy_uri);
1086     } else if (resolver_i == 2) {
1087       EXPECT_EQ("foo:99", proxy_uri);
1088     } else if (resolver_i == 3) {
1089       EXPECT_EQ("166.155.144.33:",
1090                 proxy_uri.substr(0, proxy_uri.find(':') + 1));
1091     } else {
1092       NOTREACHED();
1093     }
1094   }
1095 }
1096 
1097 }  // namespace
1098 
1099 }  // namespace net
1100