// Copyright (c) 2012 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 "chrome/browser/net/connection_tester.h"
#include "base/prefs/testing_pref_service.h"
#include "content/public/test/test_browser_thread.h"
#include "net/cert/mock_cert_verifier.h"
#include "net/cookies/cookie_monster.h"
#include "net/dns/mock_host_resolver.h"
#include "net/ftp/ftp_network_layer.h"
#include "net/http/http_auth_handler_factory.h"
#include "net/http/http_network_layer.h"
#include "net/http/http_network_session.h"
#include "net/http/http_server_properties_impl.h"
#include "net/http/transport_security_state.h"
#include "net/proxy/proxy_config_service_fixed.h"
#include "net/proxy/proxy_service.h"
#include "net/ssl/ssl_config_service_defaults.h"
#include "net/test/spawned_test_server/spawned_test_server.h"
#include "net/url_request/url_request_context.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "testing/platform_test.h"
using content::BrowserThread;
namespace {
// This is a testing delegate which simply counts how many times each of
// the delegate's methods were invoked.
class ConnectionTesterDelegate : public ConnectionTester::Delegate {
public:
ConnectionTesterDelegate()
: start_connection_test_suite_count_(0),
start_connection_test_experiment_count_(0),
completed_connection_test_experiment_count_(0),
completed_connection_test_suite_count_(0) {
}
virtual void OnStartConnectionTestSuite() OVERRIDE {
start_connection_test_suite_count_++;
}
virtual void OnStartConnectionTestExperiment(
const ConnectionTester::Experiment& experiment) OVERRIDE {
start_connection_test_experiment_count_++;
}
virtual void OnCompletedConnectionTestExperiment(
const ConnectionTester::Experiment& experiment,
int result) OVERRIDE {
completed_connection_test_experiment_count_++;
}
virtual void OnCompletedConnectionTestSuite() OVERRIDE {
completed_connection_test_suite_count_++;
base::MessageLoop::current()->Quit();
}
int start_connection_test_suite_count() const {
return start_connection_test_suite_count_;
}
int start_connection_test_experiment_count() const {
return start_connection_test_experiment_count_;
}
int completed_connection_test_experiment_count() const {
return completed_connection_test_experiment_count_;
}
int completed_connection_test_suite_count() const {
return completed_connection_test_suite_count_;
}
private:
int start_connection_test_suite_count_;
int start_connection_test_experiment_count_;
int completed_connection_test_experiment_count_;
int completed_connection_test_suite_count_;
};
// The test fixture is responsible for:
// - Making sure each test has an IO loop running
// - Catching any host resolve requests and mapping them to localhost
// (so the test doesn't use any external network dependencies).
class ConnectionTesterTest : public PlatformTest {
public:
ConnectionTesterTest()
: message_loop_(base::MessageLoop::TYPE_IO),
io_thread_(BrowserThread::IO, &message_loop_),
test_server_(net::SpawnedTestServer::TYPE_HTTP,
net::SpawnedTestServer::kLocalhost,
// Nothing is read in this directory.
base::FilePath(FILE_PATH_LITERAL("chrome"))),
proxy_script_fetcher_context_(new net::URLRequestContext) {
InitializeRequestContext();
}
protected:
// Destroy last the MessageLoop last to give a chance for objects like
// ObserverListThreadSave to shut down properly. For example,
// SSLClientAuthCache calls RemoveObserver when destroyed, but if the
// MessageLoop is already destroyed, then the RemoveObserver will be a
// no-op, and the ObserverList will contain invalid entries.
base::MessageLoop message_loop_;
content::TestBrowserThread io_thread_;
net::SpawnedTestServer test_server_;
ConnectionTesterDelegate test_delegate_;
net::MockHostResolver host_resolver_;
scoped_ptr<net::CertVerifier> cert_verifier_;
scoped_ptr<net::TransportSecurityState> transport_security_state_;
scoped_ptr<net::ProxyService> proxy_service_;
scoped_refptr<net::SSLConfigService> ssl_config_service_;
scoped_ptr<net::HttpTransactionFactory> http_transaction_factory_;
net::HttpAuthHandlerRegistryFactory http_auth_handler_factory_;
net::HttpServerPropertiesImpl http_server_properties_impl_;
scoped_ptr<net::URLRequestContext> proxy_script_fetcher_context_;
private:
void InitializeRequestContext() {
proxy_script_fetcher_context_->set_host_resolver(&host_resolver_);
cert_verifier_.reset(new net::MockCertVerifier);
transport_security_state_.reset(new net::TransportSecurityState);
proxy_script_fetcher_context_->set_cert_verifier(cert_verifier_.get());
proxy_script_fetcher_context_->set_transport_security_state(
transport_security_state_.get());
proxy_script_fetcher_context_->set_http_auth_handler_factory(
&http_auth_handler_factory_);
proxy_service_.reset(net::ProxyService::CreateDirect());
proxy_script_fetcher_context_->set_proxy_service(proxy_service_.get());
ssl_config_service_ = new net::SSLConfigServiceDefaults;
net::HttpNetworkSession::Params session_params;
session_params.host_resolver = &host_resolver_;
session_params.cert_verifier = cert_verifier_.get();
session_params.transport_security_state = transport_security_state_.get();
session_params.http_auth_handler_factory = &http_auth_handler_factory_;
session_params.ssl_config_service = ssl_config_service_.get();
session_params.proxy_service = proxy_service_.get();
session_params.http_server_properties =
http_server_properties_impl_.GetWeakPtr();
scoped_refptr<net::HttpNetworkSession> network_session(
new net::HttpNetworkSession(session_params));
http_transaction_factory_.reset(
new net::HttpNetworkLayer(network_session.get()));
proxy_script_fetcher_context_->set_http_transaction_factory(
http_transaction_factory_.get());
// In-memory cookie store.
proxy_script_fetcher_context_->set_cookie_store(
new net::CookieMonster(NULL, NULL));
}
};
TEST_F(ConnectionTesterTest, RunAllTests) {
ASSERT_TRUE(test_server_.Start());
ConnectionTester tester(&test_delegate_,
proxy_script_fetcher_context_.get(),
NULL);
// Start the test suite on URL "echoall".
// TODO(eroman): Is this URL right?
tester.RunAllTests(test_server_.GetURL("echoall"));
// Wait for all the tests to complete.
base::MessageLoop::current()->Run();
const int kNumExperiments =
ConnectionTester::PROXY_EXPERIMENT_COUNT *
ConnectionTester::HOST_RESOLVER_EXPERIMENT_COUNT;
EXPECT_EQ(1, test_delegate_.start_connection_test_suite_count());
EXPECT_EQ(kNumExperiments,
test_delegate_.start_connection_test_experiment_count());
EXPECT_EQ(kNumExperiments,
test_delegate_.completed_connection_test_experiment_count());
EXPECT_EQ(1, test_delegate_.completed_connection_test_suite_count());
}
TEST_F(ConnectionTesterTest, DeleteWhileInProgress) {
ASSERT_TRUE(test_server_.Start());
scoped_ptr<ConnectionTester> tester(
new ConnectionTester(&test_delegate_,
proxy_script_fetcher_context_.get(),
NULL));
// Start the test suite on URL "echoall".
// TODO(eroman): Is this URL right?
tester->RunAllTests(test_server_.GetURL("echoall"));
// Don't run the message loop at all. Otherwise the experiment's request may
// complete and post a task to run the next experiment before we quit the
// message loop.
EXPECT_EQ(1, test_delegate_.start_connection_test_suite_count());
EXPECT_EQ(1, test_delegate_.start_connection_test_experiment_count());
EXPECT_EQ(0, test_delegate_.completed_connection_test_experiment_count());
EXPECT_EQ(0, test_delegate_.completed_connection_test_suite_count());
// Delete the ConnectionTester while it is in progress.
tester.reset();
// Drain the tasks on the message loop.
//
// Note that we cannot simply stop the message loop, since that will delete
// any pending tasks instead of running them. This causes a problem with
// net::ClientSocketPoolBaseHelper, since the "Group" holds a pointer
// |backup_task| that it will try to deref during the destructor, but
// depending on the order that pending tasks were deleted in, it might
// already be invalid! See http://crbug.com/43291.
base::MessageLoop::current()->PostTask(FROM_HERE,
base::MessageLoop::QuitClosure());
base::MessageLoop::current()->Run();
}
} // namespace