// Copyright (c) 2009 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 "base/file_util.h"
#include "base/string_util.h"
#include "base/path_service.h"
#include "googleurl/src/gurl.h"
#include "net/base/load_log_unittest.h"
#include "net/base/net_errors.h"
#include "net/proxy/proxy_info.h"
#include "net/proxy/proxy_resolver_js_bindings.h"
#include "net/proxy/proxy_resolver_v8.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace net {
namespace {
// Javascript bindings for ProxyResolverV8, which returns mock values.
// Each time one of the bindings is called into, we push the input into a
// list, for later verification.
class MockJSBindings : public ProxyResolverJSBindings {
public:
MockJSBindings() : my_ip_address_count(0), my_ip_address_ex_count(0) {}
virtual void Alert(const std::string& message) {
LOG(INFO) << "PAC-alert: " << message; // Helpful when debugging.
alerts.push_back(message);
}
virtual std::string MyIpAddress() {
my_ip_address_count++;
return my_ip_address_result;
}
virtual std::string MyIpAddressEx() {
my_ip_address_ex_count++;
return my_ip_address_ex_result;
}
virtual std::string DnsResolve(const std::string& host) {
dns_resolves.push_back(host);
return dns_resolve_result;
}
virtual std::string DnsResolveEx(const std::string& host) {
dns_resolves_ex.push_back(host);
return dns_resolve_ex_result;
}
virtual void OnError(int line_number, const std::string& message) {
// Helpful when debugging.
LOG(INFO) << "PAC-error: [" << line_number << "] " << message;
errors.push_back(message);
errors_line_number.push_back(line_number);
}
// Mock values to return.
std::string my_ip_address_result;
std::string my_ip_address_ex_result;
std::string dns_resolve_result;
std::string dns_resolve_ex_result;
// Inputs we got called with.
std::vector<std::string> alerts;
std::vector<std::string> errors;
std::vector<int> errors_line_number;
std::vector<std::string> dns_resolves;
std::vector<std::string> dns_resolves_ex;
int my_ip_address_count;
int my_ip_address_ex_count;
};
// This is the same as ProxyResolverV8, but it uses mock bindings in place of
// the default bindings, and has a helper function to load PAC scripts from
// disk.
class ProxyResolverV8WithMockBindings : public ProxyResolverV8 {
public:
ProxyResolverV8WithMockBindings() : ProxyResolverV8(new MockJSBindings()) {}
MockJSBindings* mock_js_bindings() const {
return reinterpret_cast<MockJSBindings*>(js_bindings());
}
// Initialize with the PAC script data at |filename|.
int SetPacScriptFromDisk(const char* filename) {
FilePath path;
PathService::Get(base::DIR_SOURCE_ROOT, &path);
path = path.AppendASCII("net");
path = path.AppendASCII("data");
path = path.AppendASCII("proxy_resolver_v8_unittest");
path = path.AppendASCII(filename);
// Try to read the file from disk.
std::string file_contents;
bool ok = file_util::ReadFileToString(path, &file_contents);
// If we can't load the file from disk, something is misconfigured.
if (!ok) {
LOG(ERROR) << "Failed to read file: " << path.value();
return ERR_UNEXPECTED;
}
// Load the PAC script into the ProxyResolver.
return SetPacScriptByData(file_contents, NULL);
}
};
// Doesn't really matter what these values are for many of the tests.
const GURL kQueryUrl("http://www.google.com");
const GURL kPacUrl;
TEST(ProxyResolverV8Test, Direct) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("direct.js");
EXPECT_EQ(OK, result);
ProxyInfo proxy_info;
scoped_refptr<LoadLog> log(new LoadLog(LoadLog::kUnbounded));
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, log);
EXPECT_EQ(OK, result);
EXPECT_TRUE(proxy_info.is_direct());
EXPECT_EQ(0U, resolver.mock_js_bindings()->alerts.size());
EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
// No bindings were called, so no log entries.
EXPECT_EQ(0u, log->entries().size());
}
TEST(ProxyResolverV8Test, ReturnEmptyString) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("return_empty_string.js");
EXPECT_EQ(OK, result);
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(OK, result);
EXPECT_TRUE(proxy_info.is_direct());
EXPECT_EQ(0U, resolver.mock_js_bindings()->alerts.size());
EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
}
TEST(ProxyResolverV8Test, Basic) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("passthrough.js");
EXPECT_EQ(OK, result);
// The "FindProxyForURL" of this PAC script simply concatenates all of the
// arguments into a pseudo-host. The purpose of this test is to verify that
// the correct arguments are being passed to FindProxyForURL().
{
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(GURL("http://query.com/path"),
&proxy_info, NULL, NULL, NULL);
EXPECT_EQ(OK, result);
EXPECT_EQ("http.query.com.path.query.com:80",
proxy_info.proxy_server().ToURI());
}
{
ProxyInfo proxy_info;
int result = resolver.GetProxyForURL(GURL("ftp://query.com:90/path"),
&proxy_info, NULL, NULL, NULL);
EXPECT_EQ(OK, result);
// Note that FindProxyForURL(url, host) does not expect |host| to contain
// the port number.
EXPECT_EQ("ftp.query.com.90.path.query.com:80",
proxy_info.proxy_server().ToURI());
EXPECT_EQ(0U, resolver.mock_js_bindings()->alerts.size());
EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
}
// We call this so we'll have code coverage of the function and valgrind will
// make sure nothing bad happens.
//
// NOTE: This is here instead of in its own test so that we'll be calling it
// after having done something, in hopes it won't be a no-op.
resolver.PurgeMemory();
}
TEST(ProxyResolverV8Test, BadReturnType) {
// These are the filenames of PAC scripts which each return a non-string
// types for FindProxyForURL(). They should all fail with
// ERR_PAC_SCRIPT_FAILED.
static const char* const filenames[] = {
"return_undefined.js",
"return_integer.js",
"return_function.js",
"return_object.js",
// TODO(eroman): Should 'null' be considered equivalent to "DIRECT" ?
"return_null.js"
};
for (size_t i = 0; i < arraysize(filenames); ++i) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk(filenames[i]);
EXPECT_EQ(OK, result);
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, result);
MockJSBindings* bindings = resolver.mock_js_bindings();
EXPECT_EQ(0U, bindings->alerts.size());
ASSERT_EQ(1U, bindings->errors.size());
EXPECT_EQ("FindProxyForURL() did not return a string.",
bindings->errors[0]);
EXPECT_EQ(-1, bindings->errors_line_number[0]);
}
}
// Try using a PAC script which defines no "FindProxyForURL" function.
TEST(ProxyResolverV8Test, NoEntryPoint) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("no_entrypoint.js");
EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, result);
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(ERR_FAILED, result);
}
// Try loading a malformed PAC script.
TEST(ProxyResolverV8Test, ParseError) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("missing_close_brace.js");
EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, result);
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(ERR_FAILED, result);
MockJSBindings* bindings = resolver.mock_js_bindings();
EXPECT_EQ(0U, bindings->alerts.size());
// We get one error during compilation.
ASSERT_EQ(1U, bindings->errors.size());
EXPECT_EQ("Uncaught SyntaxError: Unexpected end of input",
bindings->errors[0]);
EXPECT_EQ(-1, bindings->errors_line_number[0]);
}
// Run a PAC script several times, which has side-effects.
TEST(ProxyResolverV8Test, SideEffects) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("side_effects.js");
// The PAC script increments a counter each time we invoke it.
for (int i = 0; i < 3; ++i) {
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(OK, result);
EXPECT_EQ(StringPrintf("sideffect_%d:80", i),
proxy_info.proxy_server().ToURI());
}
// Reload the script -- the javascript environment should be reset, hence
// the counter starts over.
result = resolver.SetPacScriptFromDisk("side_effects.js");
EXPECT_EQ(OK, result);
for (int i = 0; i < 3; ++i) {
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(OK, result);
EXPECT_EQ(StringPrintf("sideffect_%d:80", i),
proxy_info.proxy_server().ToURI());
}
}
// Execute a PAC script which throws an exception in FindProxyForURL.
TEST(ProxyResolverV8Test, UnhandledException) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("unhandled_exception.js");
EXPECT_EQ(OK, result);
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, result);
MockJSBindings* bindings = resolver.mock_js_bindings();
EXPECT_EQ(0U, bindings->alerts.size());
ASSERT_EQ(1U, bindings->errors.size());
EXPECT_EQ("Uncaught ReferenceError: undefined_variable is not defined",
bindings->errors[0]);
EXPECT_EQ(3, bindings->errors_line_number[0]);
}
// TODO(eroman): This test is disabed right now, since the parsing of
// host/port doesn't check for non-ascii characters.
TEST(ProxyResolverV8Test, DISABLED_ReturnUnicode) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("return_unicode.js");
EXPECT_EQ(OK, result);
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
// The result from this resolve was unparseable, because it
// wasn't ascii.
EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, result);
}
// Test the PAC library functions that we expose in the JS environmnet.
TEST(ProxyResolverV8Test, JavascriptLibrary) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("pac_library_unittest.js");
EXPECT_EQ(OK, result);
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
// If the javascript side of this unit-test fails, it will throw a javascript
// exception. Otherwise it will return "PROXY success:80".
EXPECT_EQ(OK, result);
EXPECT_EQ("success:80", proxy_info.proxy_server().ToURI());
EXPECT_EQ(0U, resolver.mock_js_bindings()->alerts.size());
EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
}
// Try resolving when SetPacScriptByData() has not been called.
TEST(ProxyResolverV8Test, NoSetPacScript) {
ProxyResolverV8WithMockBindings resolver;
ProxyInfo proxy_info;
// Resolve should fail, as we are not yet initialized with a script.
int result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(ERR_FAILED, result);
// Initialize it.
result = resolver.SetPacScriptFromDisk("direct.js");
EXPECT_EQ(OK, result);
// Resolve should now succeed.
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(OK, result);
// Clear it, by initializing with an empty string.
resolver.SetPacScriptByData(std::string(), NULL);
// Resolve should fail again now.
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(ERR_FAILED, result);
// Load a good script once more.
result = resolver.SetPacScriptFromDisk("direct.js");
EXPECT_EQ(OK, result);
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(OK, result);
EXPECT_EQ(0U, resolver.mock_js_bindings()->alerts.size());
EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
}
// Test marshalling/un-marshalling of values between C++/V8.
TEST(ProxyResolverV8Test, V8Bindings) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("bindings.js");
EXPECT_EQ(OK, result);
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(OK, result);
EXPECT_TRUE(proxy_info.is_direct());
MockJSBindings* bindings = resolver.mock_js_bindings();
EXPECT_EQ(0U, resolver.mock_js_bindings()->errors.size());
// Alert was called 5 times.
ASSERT_EQ(5U, bindings->alerts.size());
EXPECT_EQ("undefined", bindings->alerts[0]);
EXPECT_EQ("null", bindings->alerts[1]);
EXPECT_EQ("undefined", bindings->alerts[2]);
EXPECT_EQ("[object Object]", bindings->alerts[3]);
EXPECT_EQ("exception from calling toString()", bindings->alerts[4]);
// DnsResolve was called 8 times.
ASSERT_EQ(8U, bindings->dns_resolves.size());
EXPECT_EQ("undefined", bindings->dns_resolves[0]);
EXPECT_EQ("null", bindings->dns_resolves[1]);
EXPECT_EQ("undefined", bindings->dns_resolves[2]);
EXPECT_EQ("", bindings->dns_resolves[3]);
EXPECT_EQ("[object Object]", bindings->dns_resolves[4]);
EXPECT_EQ("function fn() {}", bindings->dns_resolves[5]);
// TODO(eroman): This isn't quite right... should probably stringize
// to something like "['3']".
EXPECT_EQ("3", bindings->dns_resolves[6]);
EXPECT_EQ("arg1", bindings->dns_resolves[7]);
// MyIpAddress was called two times.
EXPECT_EQ(2, bindings->my_ip_address_count);
// MyIpAddressEx was called once.
EXPECT_EQ(1, bindings->my_ip_address_ex_count);
// DnsResolveEx was called 2 times.
ASSERT_EQ(2U, bindings->dns_resolves_ex.size());
EXPECT_EQ("is_resolvable", bindings->dns_resolves_ex[0]);
EXPECT_EQ("foobar", bindings->dns_resolves_ex[1]);
}
// Test that calls to the myIpAddress() and dnsResolve() bindings get
// logged to the LoadLog parameter.
TEST(ProxyResolverV8Test, LoadLog) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("simple.js");
EXPECT_EQ(OK, result);
ProxyInfo proxy_info;
scoped_refptr<LoadLog> log(new LoadLog(LoadLog::kUnbounded));
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, log);
EXPECT_EQ(OK, result);
EXPECT_FALSE(proxy_info.is_direct());
EXPECT_EQ("c:100", proxy_info.proxy_server().ToURI());
// Note that dnsResolve() was never called directly, but it appears
// in the LoadLog. This is because it gets called indirectly by
// isInNet() and isResolvable().
EXPECT_EQ(6u, log->entries().size());
EXPECT_TRUE(LogContainsBeginEvent(
*log, 0, LoadLog::TYPE_PROXY_RESOLVER_V8_MY_IP_ADDRESS));
EXPECT_TRUE(LogContainsEndEvent(
*log, 1, LoadLog::TYPE_PROXY_RESOLVER_V8_MY_IP_ADDRESS));
EXPECT_TRUE(LogContainsBeginEvent(
*log, 2, LoadLog::TYPE_PROXY_RESOLVER_V8_DNS_RESOLVE));
EXPECT_TRUE(LogContainsEndEvent(
*log, 3, LoadLog::TYPE_PROXY_RESOLVER_V8_DNS_RESOLVE));
EXPECT_TRUE(LogContainsBeginEvent(
*log, 4, LoadLog::TYPE_PROXY_RESOLVER_V8_DNS_RESOLVE));
EXPECT_TRUE(LogContainsEndEvent(
*log, 5, LoadLog::TYPE_PROXY_RESOLVER_V8_DNS_RESOLVE));
}
// Try loading a PAC script that ends with a comment and has no terminal
// newline. This should not cause problems with the PAC utility functions
// that we add to the script's environment.
// http://crbug.com/22864
TEST(ProxyResolverV8Test, EndsWithCommentNoNewline) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("ends_with_comment.js");
EXPECT_EQ(OK, result);
ProxyInfo proxy_info;
scoped_refptr<LoadLog> log(new LoadLog(LoadLog::kUnbounded));
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, log);
EXPECT_EQ(OK, result);
EXPECT_FALSE(proxy_info.is_direct());
EXPECT_EQ("success:80", proxy_info.proxy_server().ToURI());
}
// Try loading a PAC script that ends with a statement and has no terminal
// newline. This should not cause problems with the PAC utility functions
// that we add to the script's environment.
// http://crbug.com/22864
TEST(ProxyResolverV8Test, EndsWithStatementNoNewline) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk(
"ends_with_statement_no_semicolon.js");
EXPECT_EQ(OK, result);
ProxyInfo proxy_info;
scoped_refptr<LoadLog> log(new LoadLog(LoadLog::kUnbounded));
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, log);
EXPECT_EQ(OK, result);
EXPECT_FALSE(proxy_info.is_direct());
EXPECT_EQ("success:3", proxy_info.proxy_server().ToURI());
}
// Test the return values from myIpAddress(), myIpAddressEx(), dnsResolve(),
// dnsResolveEx(), isResolvable(), isResolvableEx(), when the the binding
// returns empty string (failure). This simulates the return values from
// those functions when the underlying DNS resolution fails.
TEST(ProxyResolverV8Test, DNSResolutionFailure) {
ProxyResolverV8WithMockBindings resolver;
int result = resolver.SetPacScriptFromDisk("dns_fail.js");
EXPECT_EQ(OK, result);
ProxyInfo proxy_info;
result = resolver.GetProxyForURL(kQueryUrl, &proxy_info, NULL, NULL, NULL);
EXPECT_EQ(OK, result);
EXPECT_FALSE(proxy_info.is_direct());
EXPECT_EQ("success:80", proxy_info.proxy_server().ToURI());
}
} // namespace
} // namespace net