1 // Copyright (c) 2010 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 #define LOG_TAG "ProxyResolverTest"
6 #include <utils/Log.h>
7
8 #include <string.h>
9
10 #include <gtest/gtest.h>
11
12 #include "android_runtime/AndroidRuntime.h"
13 #include "proxy_test_script.h"
14 #include "proxy_resolver_v8.h"
15
16 using namespace android;
17 namespace net {
18 namespace {
19
20 // Javascript bindings for ProxyResolverV8, which returns mock values.
21 // Each time one of the bindings is called into, we push the input into a
22 // list, for later verification.
23 class MockJSBindings : public ProxyResolverJSBindings, public ProxyErrorListener {
24 public:
MockJSBindings()25 MockJSBindings() : my_ip_address_count(0), my_ip_address_ex_count(0) {}
26
MyIpAddress(std::string * ip_address)27 virtual bool MyIpAddress(std::string* ip_address) {
28 my_ip_address_count++;
29 *ip_address = my_ip_address_result;
30 return !my_ip_address_result.empty();
31 }
32
MyIpAddressEx(std::string * ip_address_list)33 virtual bool MyIpAddressEx(std::string* ip_address_list) {
34 my_ip_address_ex_count++;
35 *ip_address_list = my_ip_address_ex_result;
36 return !my_ip_address_ex_result.empty();
37 }
38
DnsResolve(const std::string & host,std::string * ip_address)39 virtual bool DnsResolve(const std::string& host, std::string* ip_address) {
40 dns_resolves.push_back(host);
41 *ip_address = dns_resolve_result;
42 return !dns_resolve_result.empty();
43 }
44
DnsResolveEx(const std::string & host,std::string * ip_address_list)45 virtual bool DnsResolveEx(const std::string& host,
46 std::string* ip_address_list) {
47 dns_resolves_ex.push_back(host);
48 *ip_address_list = dns_resolve_ex_result;
49 return !dns_resolve_ex_result.empty();
50 }
51
AlertMessage(String16 message)52 virtual void AlertMessage(String16 message) {
53 String8 m8(message);
54 std::string mstd(m8.string());
55
56 ALOGD("PAC-alert: %s\n", mstd.c_str()); // Helpful when debugging.
57 alerts.push_back(mstd);
58 }
59
ErrorMessage(const String16 message)60 virtual void ErrorMessage(const String16 message) {
61 String8 m8(message);
62 std::string mstd(m8.string());
63
64 ALOGD("PAC-error: %s\n", mstd.c_str()); // Helpful when debugging.
65 errors.push_back(mstd);
66 }
67
Shutdown()68 virtual void Shutdown() {}
69
70 // Mock values to return.
71 std::string my_ip_address_result;
72 std::string my_ip_address_ex_result;
73 std::string dns_resolve_result;
74 std::string dns_resolve_ex_result;
75
76 // Inputs we got called with.
77 std::vector<std::string> alerts;
78 std::vector<std::string> errors;
79 std::vector<std::string> dns_resolves;
80 std::vector<std::string> dns_resolves_ex;
81 int my_ip_address_count;
82 int my_ip_address_ex_count;
83 };
84
85 // This is the same as ProxyResolverV8, but it uses mock bindings in place of
86 // the default bindings, and has a helper function to load PAC scripts from
87 // disk.
88 class ProxyResolverV8WithMockBindings : public ProxyResolverV8 {
89 public:
ProxyResolverV8WithMockBindings(MockJSBindings * mock_js_bindings)90 ProxyResolverV8WithMockBindings(MockJSBindings* mock_js_bindings) :
91 ProxyResolverV8(mock_js_bindings, mock_js_bindings), mock_js_bindings_(mock_js_bindings) {
92 }
93
mock_js_bindings() const94 MockJSBindings* mock_js_bindings() const {
95 return mock_js_bindings_;
96 }
97
98 private:
99 MockJSBindings* mock_js_bindings_;
100 };
101
102 // Doesn't really matter what these values are for many of the tests.
103 const String16 kQueryUrl("http://www.google.com");
104 const String16 kQueryHost("www.google.com");
105 String16 kResults;
106
107 String16 currentPac;
108 #define SCRIPT(x) (currentPac = String16(x))
109
addString(std::vector<std::string> * list,std::string str)110 void addString(std::vector<std::string>* list, std::string str) {
111 if (str.compare(0, 6, "DIRECT") == 0) {
112 list->push_back("DIRECT");
113 } else if (str.compare(0, 6, "PROXY ") == 0) {
114 list->push_back(str.substr(6));
115 } else {
116 ALOGE("Unrecognized proxy string");
117 }
118 }
119
string16ToProxyList(String16 response)120 std::vector<std::string> string16ToProxyList(String16 response) {
121 std::vector<std::string> ret;
122 String8 response8(response);
123 std::string rstr(response8.string());
124 if (rstr.find(';') == std::string::npos) {
125 addString(&ret, rstr);
126 return ret;
127 }
128 char str[128];
129 rstr.copy(str, 0, rstr.length());
130 const char* pch = strtok(str, ";");
131
132 while (pch != NULL) {
133 // Skip leading whitespace
134 while ((*pch) == ' ') ++pch;
135 std::string pstring(pch);
136 addString(&ret, pstring);
137
138 pch = strtok(NULL, "; \t");
139 }
140
141 return ret;
142 }
143
StringPrintf(std::string str,int d)144 std::string StringPrintf(std::string str, int d) {
145 char buf[30];
146 sprintf(buf, str.c_str(), d);
147 return std::string(buf);
148 }
149
TEST(ProxyResolverV8Test,Direct)150 TEST(ProxyResolverV8Test, Direct) {
151 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
152 int result = resolver.SetPacScript(SCRIPT(DIRECT_JS));
153 EXPECT_EQ(OK, result);
154
155 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
156
157 EXPECT_EQ(OK, result);
158 std::vector<std::string> proxies = string16ToProxyList(kResults);
159 EXPECT_EQ(proxies.size(), 1U);
160 EXPECT_EQ("DIRECT",proxies[0]);
161
162 EXPECT_EQ(0U, resolver.mock_js_bindings()->alerts.size());
163 EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
164 }
165
TEST(ProxyResolverV8Test,ReturnEmptyString)166 TEST(ProxyResolverV8Test, ReturnEmptyString) {
167 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
168 int result = resolver.SetPacScript(SCRIPT(RETURN_EMPTY_STRING_JS));
169 EXPECT_EQ(OK, result);
170
171 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
172
173 EXPECT_EQ(OK, result);
174 std::vector<std::string> proxies = string16ToProxyList(kResults);
175 EXPECT_EQ(proxies.size(), 0U);
176
177 EXPECT_EQ(0U, resolver.mock_js_bindings()->alerts.size());
178 EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
179 }
180
TEST(ProxyResolverV8Test,Basic)181 TEST(ProxyResolverV8Test, Basic) {
182 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
183 int result = resolver.SetPacScript(SCRIPT(PASSTHROUGH_JS));
184 EXPECT_EQ(OK, result);
185
186 // The "FindProxyForURL" of this PAC script simply concatenates all of the
187 // arguments into a pseudo-host. The purpose of this test is to verify that
188 // the correct arguments are being passed to FindProxyForURL().
189 {
190 String16 queryUrl("http://query.com/path");
191 String16 queryHost("query.com");
192 result = resolver.GetProxyForURL(queryUrl, queryHost, &kResults);
193 EXPECT_EQ(OK, result);
194 std::vector<std::string> proxies = string16ToProxyList(kResults);
195 EXPECT_EQ(1U, proxies.size());
196 EXPECT_EQ("http.query.com.path.query.com", proxies[0]);
197 }
198 {
199 String16 queryUrl("ftp://query.com:90/path");
200 String16 queryHost("query.com");
201 int result = resolver.GetProxyForURL(queryUrl, queryHost, &kResults);
202
203 EXPECT_EQ(OK, result);
204 // Note that FindProxyForURL(url, host) does not expect |host| to contain
205 // the port number.
206 std::vector<std::string> proxies = string16ToProxyList(kResults);
207 EXPECT_EQ(1U, proxies.size());
208 EXPECT_EQ("ftp.query.com.90.path.query.com", proxies[0]);
209
210 EXPECT_EQ(0U, resolver.mock_js_bindings()->alerts.size());
211 EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
212 }
213
214 // We call this so we'll have code coverage of the function and valgrind will
215 // make sure nothing bad happens.
216 //
217 // NOTE: This is here instead of in its own test so that we'll be calling it
218 // after having done something, in hopes it won't be a no-op.
219 resolver.PurgeMemory();
220 }
221
TEST(ProxyResolverV8Test,BadReturnType)222 TEST(ProxyResolverV8Test, BadReturnType) {
223 // These are the files of PAC scripts which each return a non-string
224 // types for FindProxyForURL(). They should all fail with
225 // ERR_PAC_SCRIPT_FAILED.
226 static const String16 files[] = {
227 String16(RETURN_UNDEFINED_JS),
228 String16(RETURN_INTEGER_JS),
229 String16(RETURN_FUNCTION_JS),
230 String16(RETURN_OBJECT_JS),
231 String16(RETURN_NULL_JS)
232 };
233
234 for (size_t i = 0; i < 5; ++i) {
235 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
236 int result = resolver.SetPacScript(files[i]);
237 EXPECT_EQ(OK, result);
238
239 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
240
241 EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, result);
242
243 MockJSBindings* bindings = resolver.mock_js_bindings();
244 EXPECT_EQ(0U, bindings->alerts.size());
245 ASSERT_EQ(1U, bindings->errors.size());
246 EXPECT_EQ("FindProxyForURL() did not return a string.",
247 bindings->errors[0]);
248 }
249 }
250
251 // Try using a PAC script which defines no "FindProxyForURL" function.
TEST(ProxyResolverV8Test,NoEntryPoint)252 TEST(ProxyResolverV8Test, NoEntryPoint) {
253 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
254 int result = resolver.SetPacScript(SCRIPT(NO_ENTRYPOINT_JS));
255 EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, result);
256
257 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
258
259 EXPECT_EQ(ERR_FAILED, result);
260 }
261
262 // Try loading a malformed PAC script.
TEST(ProxyResolverV8Test,ParseError)263 TEST(ProxyResolverV8Test, ParseError) {
264 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
265 int result = resolver.SetPacScript(SCRIPT(MISSING_CLOSE_BRACE_JS));
266 EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, result);
267
268 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
269
270 EXPECT_EQ(ERR_FAILED, result);
271
272 MockJSBindings* bindings = resolver.mock_js_bindings();
273 EXPECT_EQ(0U, bindings->alerts.size());
274
275 // We get one error during compilation.
276 ASSERT_EQ(1U, bindings->errors.size());
277
278 EXPECT_EQ("Uncaught SyntaxError: Unexpected end of input",
279 bindings->errors[0]);
280 }
281
282 // Run a PAC script several times, which has side-effects.
TEST(ProxyResolverV8Test,SideEffects)283 TEST(ProxyResolverV8Test, SideEffects) {
284 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
285 int result = resolver.SetPacScript(SCRIPT(SIDE_EFFECTS_JS));
286
287 // The PAC script increments a counter each time we invoke it.
288 for (int i = 0; i < 3; ++i) {
289 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
290 EXPECT_EQ(OK, result);
291 std::vector<std::string> proxies = string16ToProxyList(kResults);
292 EXPECT_EQ(1U, proxies.size());
293 EXPECT_EQ(StringPrintf("sideffect_%d", i),
294 proxies[0]);
295 }
296
297 // Reload the script -- the javascript environment should be reset, hence
298 // the counter starts over.
299 result = resolver.SetPacScript(SCRIPT(SIDE_EFFECTS_JS));
300 EXPECT_EQ(OK, result);
301
302 for (int i = 0; i < 3; ++i) {
303 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
304 EXPECT_EQ(OK, result);
305 std::vector<std::string> proxies = string16ToProxyList(kResults);
306 EXPECT_EQ(1U, proxies.size());
307 EXPECT_EQ(StringPrintf("sideffect_%d", i),
308 proxies[0]);
309 }
310 }
311
312 // Execute a PAC script which throws an exception in FindProxyForURL.
TEST(ProxyResolverV8Test,UnhandledException)313 TEST(ProxyResolverV8Test, UnhandledException) {
314 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
315 int result = resolver.SetPacScript(SCRIPT(UNHANDLED_EXCEPTION_JS));
316 EXPECT_EQ(OK, result);
317
318 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
319
320 EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, result);
321
322 MockJSBindings* bindings = resolver.mock_js_bindings();
323 EXPECT_EQ(0U, bindings->alerts.size());
324 ASSERT_EQ(1U, bindings->errors.size());
325 EXPECT_EQ("Uncaught ReferenceError: undefined_variable is not defined",
326 bindings->errors[0]);
327 }
328
TEST(ProxyResolverV8Test,ReturnUnicode)329 TEST(ProxyResolverV8Test, ReturnUnicode) {
330 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
331 int result = resolver.SetPacScript(SCRIPT(RETURN_UNICODE_JS));
332 EXPECT_EQ(OK, result);
333
334 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
335
336 // The result from this resolve was unparseable, because it
337 // wasn't ASCII.
338 EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, result);
339 }
340
341 // Test the PAC library functions that we expose in the JS environmnet.
TEST(ProxyResolverV8Test,JavascriptLibrary)342 TEST(ProxyResolverV8Test, JavascriptLibrary) {
343 ALOGE("Javascript start");
344 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
345 int result = resolver.SetPacScript(SCRIPT(PAC_LIBRARY_UNITTEST_JS));
346 EXPECT_EQ(OK, result);
347
348 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
349
350 // If the javascript side of this unit-test fails, it will throw a javascript
351 // exception. Otherwise it will return "PROXY success:80".
352 EXPECT_EQ(OK, result);
353 std::vector<std::string> proxies = string16ToProxyList(kResults);
354 EXPECT_EQ(1U, proxies.size());
355 EXPECT_EQ("success:80", proxies[0]);
356
357 EXPECT_EQ(0U, resolver.mock_js_bindings()->alerts.size());
358 EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
359 }
360
361 // Try resolving when SetPacScriptByData() has not been called.
TEST(ProxyResolverV8Test,NoSetPacScript)362 TEST(ProxyResolverV8Test, NoSetPacScript) {
363 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
364
365
366 // Resolve should fail, as we are not yet initialized with a script.
367 int result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
368 EXPECT_EQ(ERR_FAILED, result);
369
370 // Initialize it.
371 result = resolver.SetPacScript(SCRIPT(DIRECT_JS));
372 EXPECT_EQ(OK, result);
373
374 // Resolve should now succeed.
375 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
376 EXPECT_EQ(OK, result);
377
378 // Clear it, by initializing with an empty string.
379 resolver.SetPacScript(SCRIPT());
380
381 // Resolve should fail again now.
382 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
383 EXPECT_EQ(ERR_FAILED, result);
384
385 // Load a good script once more.
386 result = resolver.SetPacScript(SCRIPT(DIRECT_JS));
387 EXPECT_EQ(OK, result);
388 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
389 EXPECT_EQ(OK, result);
390
391 EXPECT_EQ(0U, resolver.mock_js_bindings()->alerts.size());
392 EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
393 }
394
395 // Test marshalling/un-marshalling of values between C++/V8.
TEST(ProxyResolverV8Test,V8Bindings)396 TEST(ProxyResolverV8Test, V8Bindings) {
397 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
398 MockJSBindings* bindings = resolver.mock_js_bindings();
399 bindings->dns_resolve_result = "127.0.0.1";
400 int result = resolver.SetPacScript(SCRIPT(BINDINGS_JS));
401 EXPECT_EQ(OK, result);
402
403 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
404
405 EXPECT_EQ(OK, result);
406 std::vector<std::string> proxies = string16ToProxyList(kResults);
407 EXPECT_EQ(1U, proxies.size());
408 EXPECT_EQ("DIRECT", proxies[0]);
409
410 EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
411
412 // Alert was called 5 times.
413 ASSERT_EQ(5U, bindings->alerts.size());
414 EXPECT_EQ("undefined", bindings->alerts[0]);
415 EXPECT_EQ("null", bindings->alerts[1]);
416 EXPECT_EQ("undefined", bindings->alerts[2]);
417 EXPECT_EQ("[object Object]", bindings->alerts[3]);
418 EXPECT_EQ("exception from calling toString()", bindings->alerts[4]);
419
420 // DnsResolve was called 8 times, however only 2 of those were string
421 // parameters. (so 6 of them failed immediately).
422 ASSERT_EQ(2U, bindings->dns_resolves.size());
423 EXPECT_EQ("", bindings->dns_resolves[0]);
424 EXPECT_EQ("arg1", bindings->dns_resolves[1]);
425
426 // MyIpAddress was called two times.
427 EXPECT_EQ(2, bindings->my_ip_address_count);
428
429 // MyIpAddressEx was called once.
430 EXPECT_EQ(1, bindings->my_ip_address_ex_count);
431
432 // DnsResolveEx was called 2 times.
433 ASSERT_EQ(2U, bindings->dns_resolves_ex.size());
434 EXPECT_EQ("is_resolvable", bindings->dns_resolves_ex[0]);
435 EXPECT_EQ("foobar", bindings->dns_resolves_ex[1]);
436 }
437
438 // Test calling a binding (myIpAddress()) from the script's global scope.
439 // http://crbug.com/40026
TEST(ProxyResolverV8Test,BindingCalledDuringInitialization)440 TEST(ProxyResolverV8Test, BindingCalledDuringInitialization) {
441 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
442
443 int result = resolver.SetPacScript(SCRIPT(BINDING_FROM_GLOBAL_JS));
444 EXPECT_EQ(OK, result);
445
446 MockJSBindings* bindings = resolver.mock_js_bindings();
447
448 // myIpAddress() got called during initialization of the script.
449 EXPECT_EQ(1, bindings->my_ip_address_count);
450
451 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
452
453 EXPECT_EQ(OK, result);
454 std::vector<std::string> proxies = string16ToProxyList(kResults);
455 EXPECT_EQ(1U, proxies.size());
456 EXPECT_NE("DIRECT", proxies[0]);
457 EXPECT_EQ("127.0.0.1:80", proxies[0]);
458
459 // Check that no other bindings were called.
460 EXPECT_EQ(0U, bindings->errors.size());
461 ASSERT_EQ(0U, bindings->alerts.size());
462 ASSERT_EQ(0U, bindings->dns_resolves.size());
463 EXPECT_EQ(0, bindings->my_ip_address_ex_count);
464 ASSERT_EQ(0U, bindings->dns_resolves_ex.size());
465 }
466
467 // Try loading a PAC script that ends with a comment and has no terminal
468 // newline. This should not cause problems with the PAC utility functions
469 // that we add to the script's environment.
470 // http://crbug.com/22864
TEST(ProxyResolverV8Test,EndsWithCommentNoNewline)471 TEST(ProxyResolverV8Test, EndsWithCommentNoNewline) {
472 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
473 int result = resolver.SetPacScript(SCRIPT(ENDS_WITH_COMMENT_JS));
474 EXPECT_EQ(OK, result);
475
476 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
477
478 EXPECT_EQ(OK, result);
479 std::vector<std::string> proxies = string16ToProxyList(kResults);
480 EXPECT_EQ(1U, proxies.size());
481 EXPECT_NE("DIRECT", proxies[0]);
482 EXPECT_EQ("success:80", proxies[0]);
483 }
484
485 // Try loading a PAC script that ends with a statement and has no terminal
486 // newline. This should not cause problems with the PAC utility functions
487 // that we add to the script's environment.
488 // http://crbug.com/22864
TEST(ProxyResolverV8Test,EndsWithStatementNoNewline)489 TEST(ProxyResolverV8Test, EndsWithStatementNoNewline) {
490 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
491 int result = resolver.SetPacScript(
492 SCRIPT(ENDS_WITH_STATEMENT_NO_SEMICOLON_JS));
493 EXPECT_EQ(OK, result);
494
495 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
496
497 EXPECT_EQ(OK, result);
498 std::vector<std::string> proxies = string16ToProxyList(kResults);
499 EXPECT_EQ(1U, proxies.size());
500 EXPECT_NE("DIRECT", proxies[0]);
501 EXPECT_EQ("success:3", proxies[0]);
502 }
503
504 // Test the return values from myIpAddress(), myIpAddressEx(), dnsResolve(),
505 // dnsResolveEx(), isResolvable(), isResolvableEx(), when the the binding
506 // returns empty string (failure). This simulates the return values from
507 // those functions when the underlying DNS resolution fails.
TEST(ProxyResolverV8Test,DNSResolutionFailure)508 TEST(ProxyResolverV8Test, DNSResolutionFailure) {
509 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
510 int result = resolver.SetPacScript(SCRIPT(DNS_FAIL_JS));
511 EXPECT_EQ(OK, result);
512
513 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
514
515 EXPECT_EQ(OK, result);
516 std::vector<std::string> proxies = string16ToProxyList(kResults);
517 EXPECT_EQ(1U, proxies.size());
518 EXPECT_NE("DIRECT", proxies[0]);
519 EXPECT_EQ("success:80", proxies[0]);
520 }
521
TEST(ProxyResolverV8Test,DNSResolutionOfInternationDomainName)522 TEST(ProxyResolverV8Test, DNSResolutionOfInternationDomainName) {
523 return;
524 ProxyResolverV8WithMockBindings resolver(new MockJSBindings());
525 int result = resolver.SetPacScript(String16(INTERNATIONAL_DOMAIN_NAMES_JS));
526 EXPECT_EQ(OK, result);
527
528 // Execute FindProxyForURL().
529 result = resolver.GetProxyForURL(kQueryUrl, kQueryHost, &kResults);
530
531 EXPECT_EQ(OK, result);
532 std::vector<std::string> proxies = string16ToProxyList(kResults);
533 EXPECT_EQ(1U, proxies.size());
534 EXPECT_EQ("DIRECT", proxies[0]);
535
536 // Check that the international domain name was converted to punycode
537 // before passing it onto the bindings layer.
538 MockJSBindings* bindings = resolver.mock_js_bindings();
539
540 ASSERT_EQ(1u, bindings->dns_resolves.size());
541 EXPECT_EQ("xn--bcher-kva.ch", bindings->dns_resolves[0]);
542
543 ASSERT_EQ(1u, bindings->dns_resolves_ex.size());
544 EXPECT_EQ("xn--bcher-kva.ch", bindings->dns_resolves_ex[0]);
545 }
546
547 } // namespace
548 } // namespace net
549