// Copyright 2015 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 "net/proxy/mojo_proxy_resolver_factory_impl.h" #include "base/strings/utf_string_conversions.h" #include "net/base/test_completion_callback.h" #include "net/proxy/mock_proxy_resolver.h" #include "net/proxy/proxy_resolver_v8_tracing.h" #include "net/test/event_waiter.h" #include "testing/gtest/include/gtest/gtest.h" #include "third_party/mojo/src/mojo/public/cpp/bindings/binding.h" namespace net { namespace { const char kScriptData[] = "FooBarBaz"; class FakeProxyResolver : public ProxyResolverV8Tracing { public: explicit FakeProxyResolver(const base::Closure& on_destruction) : on_destruction_(on_destruction) {} ~FakeProxyResolver() override { on_destruction_.Run(); } private: // ProxyResolverV8Tracing overrides. void GetProxyForURL(const GURL& url, ProxyInfo* results, const CompletionCallback& callback, ProxyResolver::RequestHandle* request, scoped_ptr bindings) override {} void CancelRequest(ProxyResolver::RequestHandle request) override {} LoadState GetLoadState(ProxyResolver::RequestHandle request) const override { return LOAD_STATE_RESOLVING_PROXY_FOR_URL; } const base::Closure on_destruction_; }; enum Event { NONE, RESOLVER_CREATED, CONNECTION_ERROR, RESOLVER_DESTROYED, }; class TestProxyResolverFactory : public ProxyResolverV8TracingFactory { public: struct PendingRequest { scoped_ptr* resolver; CompletionCallback callback; }; explicit TestProxyResolverFactory(EventWaiter* waiter) : waiter_(waiter) {} void CreateProxyResolverV8Tracing( const scoped_refptr& pac_script, scoped_ptr bindings, scoped_ptr* resolver, const CompletionCallback& callback, scoped_ptr* request) override { requests_handled_++; waiter_->NotifyEvent(RESOLVER_CREATED); EXPECT_EQ(base::ASCIIToUTF16(kScriptData), pac_script->utf16()); EXPECT_TRUE(resolver); pending_request_.reset(new PendingRequest); pending_request_->resolver = resolver; pending_request_->callback = callback; ASSERT_TRUE(bindings); bindings->Alert(base::ASCIIToUTF16("alert")); bindings->OnError(10, base::ASCIIToUTF16("error")); EXPECT_TRUE(bindings->GetHostResolver()); } size_t requests_handled() { return requests_handled_; } const PendingRequest* pending_request() { return pending_request_.get(); } private: EventWaiter* waiter_; size_t requests_handled_ = 0; scoped_ptr pending_request_; }; } // namespace class MojoProxyResolverFactoryImplTest : public testing::Test, public interfaces::ProxyResolverFactoryRequestClient { public: void SetUp() override { mock_factory_ = new TestProxyResolverFactory(&waiter_); new MojoProxyResolverFactoryImpl(make_scoped_ptr(mock_factory_), mojo::GetProxy(&factory_)); } void OnConnectionError() { waiter_.NotifyEvent(CONNECTION_ERROR); } void OnFakeProxyInstanceDestroyed() { instances_destroyed_++; waiter_.NotifyEvent(RESOLVER_DESTROYED); } void ReportResult(int32_t error) override { create_callback_.Run(error); } void Alert(const mojo::String& message) override {} void OnError(int32_t line_number, const mojo::String& message) override {} void ResolveDns(interfaces::HostResolverRequestInfoPtr request_info, interfaces::HostResolverRequestClientPtr client) override {} protected: scoped_ptr mock_factory_owner_; TestProxyResolverFactory* mock_factory_; interfaces::ProxyResolverFactoryPtr factory_; int instances_destroyed_ = 0; CompletionCallback create_callback_; EventWaiter waiter_; }; TEST_F(MojoProxyResolverFactoryImplTest, DisconnectProxyResolverClient) { interfaces::ProxyResolverPtr proxy_resolver; interfaces::ProxyResolverFactoryRequestClientPtr client_ptr; mojo::Binding client_binding( this, mojo::GetProxy(&client_ptr)); factory_->CreateResolver(mojo::String::From(kScriptData), mojo::GetProxy(&proxy_resolver), client_ptr.Pass()); proxy_resolver.set_connection_error_handler( base::Bind(&MojoProxyResolverFactoryImplTest::OnConnectionError, base::Unretained(this))); waiter_.WaitForEvent(RESOLVER_CREATED); EXPECT_EQ(0, instances_destroyed_); ASSERT_EQ(1u, mock_factory_->requests_handled()); TestCompletionCallback create_callback; create_callback_ = create_callback.callback(); ASSERT_TRUE(mock_factory_->pending_request()); mock_factory_->pending_request()->resolver->reset( new FakeProxyResolver(base::Bind( &MojoProxyResolverFactoryImplTest::OnFakeProxyInstanceDestroyed, base::Unretained(this)))); mock_factory_->pending_request()->callback.Run(OK); EXPECT_EQ(OK, create_callback.WaitForResult()); proxy_resolver.reset(); waiter_.WaitForEvent(RESOLVER_DESTROYED); EXPECT_EQ(1, instances_destroyed_); } TEST_F(MojoProxyResolverFactoryImplTest, Error) { interfaces::ProxyResolverPtr proxy_resolver; interfaces::ProxyResolverFactoryRequestClientPtr client_ptr; mojo::Binding client_binding( this, mojo::GetProxy(&client_ptr)); factory_->CreateResolver(mojo::String::From(kScriptData), mojo::GetProxy(&proxy_resolver), client_ptr.Pass()); proxy_resolver.set_connection_error_handler( base::Bind(&MojoProxyResolverFactoryImplTest::OnConnectionError, base::Unretained(this))); waiter_.WaitForEvent(RESOLVER_CREATED); EXPECT_EQ(0, instances_destroyed_); ASSERT_EQ(1u, mock_factory_->requests_handled()); TestCompletionCallback create_callback; create_callback_ = create_callback.callback(); ASSERT_TRUE(mock_factory_->pending_request()); mock_factory_->pending_request()->callback.Run(ERR_PAC_SCRIPT_FAILED); EXPECT_EQ(ERR_PAC_SCRIPT_FAILED, create_callback.WaitForResult()); } TEST_F(MojoProxyResolverFactoryImplTest, DisconnectClientDuringResolverCreation) { interfaces::ProxyResolverPtr proxy_resolver; interfaces::ProxyResolverFactoryRequestClientPtr client_ptr; mojo::Binding client_binding( this, mojo::GetProxy(&client_ptr)); factory_->CreateResolver(mojo::String::From(kScriptData), mojo::GetProxy(&proxy_resolver), client_ptr.Pass()); proxy_resolver.set_connection_error_handler( base::Bind(&MojoProxyResolverFactoryImplTest::OnConnectionError, base::Unretained(this))); waiter_.WaitForEvent(RESOLVER_CREATED); EXPECT_EQ(0, instances_destroyed_); ASSERT_EQ(1u, mock_factory_->requests_handled()); client_binding.Close(); waiter_.WaitForEvent(CONNECTION_ERROR); } TEST_F(MojoProxyResolverFactoryImplTest, DisconnectFactoryDuringResolverCreation) { interfaces::ProxyResolverPtr proxy_resolver; interfaces::ProxyResolverFactoryRequestClientPtr client_ptr; mojo::Binding client_binding( this, mojo::GetProxy(&client_ptr)); factory_->CreateResolver(mojo::String::From(kScriptData), mojo::GetProxy(&proxy_resolver), client_ptr.Pass()); proxy_resolver.set_connection_error_handler( base::Bind(&MojoProxyResolverFactoryImplTest::OnConnectionError, base::Unretained(this))); client_binding.set_connection_error_handler( base::Bind(&MojoProxyResolverFactoryImplTest::OnConnectionError, base::Unretained(this))); waiter_.WaitForEvent(RESOLVER_CREATED); EXPECT_EQ(0, instances_destroyed_); ASSERT_EQ(1u, mock_factory_->requests_handled()); factory_.reset(); waiter_.WaitForEvent(CONNECTION_ERROR); } } // namespace net