// Copyright (c) 2011 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 #include #include #include "base/message_loop.h" #include "base/threading/thread.h" #include "base/synchronization/waitable_event.h" #include "net/base/net_errors.h" #include "net/http/http_response_headers.h" #include "net/url_request/url_request.h" #include "net/url_request/url_request_error_job.h" #include "testing/gtest/include/gtest/gtest.h" #include "webkit/appcache/appcache.h" #include "webkit/appcache/appcache_backend_impl.h" #include "webkit/appcache/appcache_request_handler.h" #include "webkit/appcache/appcache_url_request_job.h" #include "webkit/appcache/mock_appcache_policy.h" #include "webkit/appcache/mock_appcache_service.h" namespace appcache { static const int kMockProcessId = 1; class AppCacheRequestHandlerTest : public testing::Test { public: class MockFrontend : public AppCacheFrontend { public: virtual void OnCacheSelected( int host_id, const appcache::AppCacheInfo& info) {} virtual void OnStatusChanged(const std::vector& host_ids, appcache::Status status) {} virtual void OnEventRaised(const std::vector& host_ids, appcache::EventID event_id) {} virtual void OnErrorEventRaised(const std::vector& host_ids, const std::string& message) {} virtual void OnProgressEventRaised(const std::vector& host_ids, const GURL& url, int num_total, int num_complete) {} virtual void OnLogMessage(int host_id, appcache::LogLevel log_level, const std::string& message) {} virtual void OnContentBlocked(int host_id, const GURL& manifest_url) {} }; // Helper class run a test on our io_thread. The io_thread // is spun up once and reused for all tests. template class WrapperTask : public Task { public: WrapperTask(AppCacheRequestHandlerTest* test, Method method) : test_(test), method_(method) { } virtual void Run() { test_->SetUpTest(); (test_->*method_)(); } private: AppCacheRequestHandlerTest* test_; Method method_; }; // Subclasses to simulate particular responses so test cases can // exercise fallback code paths. class MockURLRequestDelegate : public net::URLRequest::Delegate { virtual void OnResponseStarted(net::URLRequest* request) {} virtual void OnReadCompleted(net::URLRequest* request, int bytes_read) {} }; class MockURLRequestJob : public net::URLRequestJob { public: MockURLRequestJob( net::URLRequest* request, int response_code) : net::URLRequestJob(request), response_code_(response_code), has_response_info_(false) {} MockURLRequestJob( net::URLRequest* request, const net::HttpResponseInfo& info) : net::URLRequestJob(request), response_code_(info.headers->response_code()), has_response_info_(true), response_info_(info) {} virtual void Start() { NotifyHeadersComplete(); } virtual int GetResponseCode() const { return response_code_; } virtual void GetResponseInfo(net::HttpResponseInfo* info) { if (!has_response_info_) return; *info = response_info_; } int response_code_; bool has_response_info_; net::HttpResponseInfo response_info_; }; class MockURLRequest : public net::URLRequest { public: explicit MockURLRequest(const GURL& url) : net::URLRequest(url, NULL) {} void SimulateResponseCode(int http_response_code) { mock_factory_job_ = new MockURLRequestJob(this, http_response_code); Start(); DCHECK(!mock_factory_job_); // All our simulation needs to satisfy are the following two DCHECKs DCHECK(status().is_success()); DCHECK_EQ(http_response_code, GetResponseCode()); } void SimulateResponseInfo(const net::HttpResponseInfo& info) { mock_factory_job_ = new MockURLRequestJob(this, info); set_delegate(&delegate_); // needed to get the info back out Start(); DCHECK(!mock_factory_job_); } MockURLRequestDelegate delegate_; }; static net::URLRequestJob* MockHttpJobFactory(net::URLRequest* request, const std::string& scheme) { if (mock_factory_job_) { net::URLRequestJob* temp = mock_factory_job_; mock_factory_job_ = NULL; return temp; } else { // Some of these tests trigger UpdateJobs which start URLRequests. // We short circuit those be returning error jobs. return new net::URLRequestErrorJob(request, net::ERR_INTERNET_DISCONNECTED); } } static void SetUpTestCase() { io_thread_.reset(new base::Thread("AppCacheRequestHandlerTest Thread")); base::Thread::Options options(MessageLoop::TYPE_IO, 0); io_thread_->StartWithOptions(options); } static void TearDownTestCase() { io_thread_.reset(NULL); } // Test harness -------------------------------------------------- AppCacheRequestHandlerTest() : host_(NULL), orig_http_factory_(NULL) { } template void RunTestOnIOThread(Method method) { test_finished_event_ .reset(new base::WaitableEvent(false, false)); io_thread_->message_loop()->PostTask( FROM_HERE, new WrapperTask(this, method)); test_finished_event_->Wait(); } void SetUpTest() { DCHECK(MessageLoop::current() == io_thread_->message_loop()); orig_http_factory_ = net::URLRequest::Deprecated::RegisterProtocolFactory( "http", MockHttpJobFactory); mock_service_.reset(new MockAppCacheService); mock_policy_.reset(new MockAppCachePolicy); mock_service_->set_appcache_policy(mock_policy_.get()); mock_frontend_.reset(new MockFrontend); backend_impl_.reset(new AppCacheBackendImpl); backend_impl_->Initialize(mock_service_.get(), mock_frontend_.get(), kMockProcessId); const int kHostId = 1; backend_impl_->RegisterHost(kHostId); host_ = backend_impl_->GetHost(kHostId); } void TearDownTest() { DCHECK(MessageLoop::current() == io_thread_->message_loop()); DCHECK(!mock_factory_job_); net::URLRequest::Deprecated::RegisterProtocolFactory( "http", orig_http_factory_); orig_http_factory_ = NULL; job_ = NULL; handler_.reset(); request_.reset(); backend_impl_.reset(); mock_frontend_.reset(); mock_service_.reset(); mock_policy_.reset(); host_ = NULL; } void TestFinished() { // We unwind the stack prior to finishing up to let stack // based objects get deleted. DCHECK(MessageLoop::current() == io_thread_->message_loop()); MessageLoop::current()->PostTask(FROM_HERE, NewRunnableMethod( this, &AppCacheRequestHandlerTest::TestFinishedUnwound)); } void TestFinishedUnwound() { TearDownTest(); test_finished_event_->Signal(); } void PushNextTask(Task* task) { task_stack_.push(task); } void ScheduleNextTask() { DCHECK(MessageLoop::current() == io_thread_->message_loop()); if (task_stack_.empty()) { TestFinished(); return; } MessageLoop::current()->PostTask(FROM_HERE, task_stack_.top()); task_stack_.pop(); } // MainResource_Miss -------------------------------------------------- void MainResource_Miss() { PushNextTask(NewRunnableMethod( this, &AppCacheRequestHandlerTest::Verify_MainResource_Miss)); request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::MAIN_FRAME)); EXPECT_TRUE(handler_.get()); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_TRUE(job_.get()); EXPECT_TRUE(job_->is_waiting()); // We have to wait for completion of storage->FindResponseForMainRequest. ScheduleNextTask(); } void Verify_MainResource_Miss() { EXPECT_FALSE(job_->is_waiting()); EXPECT_TRUE(job_->is_delivering_network_response()); int64 cache_id = kNoCacheId; GURL manifest_url; handler_->GetExtraResponseInfo(&cache_id, &manifest_url); EXPECT_EQ(kNoCacheId, cache_id); EXPECT_EQ(GURL(), manifest_url); AppCacheURLRequestJob* fallback_job; fallback_job = handler_->MaybeLoadFallbackForRedirect( request_.get(), GURL("http://blah/redirect")); EXPECT_FALSE(fallback_job); fallback_job = handler_->MaybeLoadFallbackForResponse(request_.get()); EXPECT_FALSE(fallback_job); EXPECT_TRUE(host_->preferred_manifest_url().is_empty()); TestFinished(); } // MainResource_Hit -------------------------------------------------- void MainResource_Hit() { PushNextTask(NewRunnableMethod( this, &AppCacheRequestHandlerTest::Verify_MainResource_Hit)); request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::MAIN_FRAME)); EXPECT_TRUE(handler_.get()); mock_storage()->SimulateFindMainResource( AppCacheEntry(AppCacheEntry::EXPLICIT, 1), GURL(), AppCacheEntry(), 1, GURL("http://blah/manifest/")); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_TRUE(job_.get()); EXPECT_TRUE(job_->is_waiting()); // We have to wait for completion of storage->FindResponseForMainRequest. ScheduleNextTask(); } void Verify_MainResource_Hit() { EXPECT_FALSE(job_->is_waiting()); EXPECT_TRUE(job_->is_delivering_appcache_response()); int64 cache_id = kNoCacheId; GURL manifest_url; handler_->GetExtraResponseInfo(&cache_id, &manifest_url); EXPECT_EQ(1, cache_id); EXPECT_EQ(GURL("http://blah/manifest/"), manifest_url); AppCacheURLRequestJob* fallback_job; fallback_job = handler_->MaybeLoadFallbackForResponse(request_.get()); EXPECT_FALSE(fallback_job); EXPECT_EQ(GURL("http://blah/manifest/"), host_->preferred_manifest_url()); TestFinished(); } // MainResource_Fallback -------------------------------------------------- void MainResource_Fallback() { PushNextTask(NewRunnableMethod( this, &AppCacheRequestHandlerTest::Verify_MainResource_Fallback)); request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::MAIN_FRAME)); EXPECT_TRUE(handler_.get()); mock_storage()->SimulateFindMainResource( AppCacheEntry(), GURL("http://blah/fallbackurl"), AppCacheEntry(AppCacheEntry::EXPLICIT, 1), 1, GURL("http://blah/manifest/")); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_TRUE(job_.get()); EXPECT_TRUE(job_->is_waiting()); // We have to wait for completion of storage->FindResponseForMainRequest. ScheduleNextTask(); } void Verify_MainResource_Fallback() { EXPECT_FALSE(job_->is_waiting()); EXPECT_TRUE(job_->is_delivering_network_response()); // When the request is restarted, the existing job is dropped so a // real network job gets created. We expect NULL here which will cause // the net library to create a real job. job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_FALSE(job_); // Simulate an http error of the real network job. request_->SimulateResponseCode(500); job_ = handler_->MaybeLoadFallbackForResponse(request_.get()); EXPECT_TRUE(job_); EXPECT_TRUE(job_->is_delivering_appcache_response()); int64 cache_id = kNoCacheId; GURL manifest_url; handler_->GetExtraResponseInfo(&cache_id, &manifest_url); EXPECT_EQ(1, cache_id); EXPECT_EQ(GURL("http://blah/manifest/"), manifest_url); EXPECT_TRUE(host_->main_resource_was_fallback_); EXPECT_EQ(GURL("http://blah/fallbackurl"), host_->fallback_url_); EXPECT_EQ(GURL("http://blah/manifest/"), host_->preferred_manifest_url()); TestFinished(); } // MainResource_FallbackOverride -------------------------------------------- void MainResource_FallbackOverride() { PushNextTask(NewRunnableMethod( this, &AppCacheRequestHandlerTest::Verify_MainResource_FallbackOverride)); request_.reset(new MockURLRequest(GURL("http://blah/fallback-override"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::MAIN_FRAME)); EXPECT_TRUE(handler_.get()); mock_storage()->SimulateFindMainResource( AppCacheEntry(), GURL("http://blah/fallbackurl"), AppCacheEntry(AppCacheEntry::EXPLICIT, 1), 1, GURL("http://blah/manifest/")); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_TRUE(job_.get()); EXPECT_TRUE(job_->is_waiting()); // We have to wait for completion of storage->FindResponseForMainRequest. ScheduleNextTask(); } void Verify_MainResource_FallbackOverride() { EXPECT_FALSE(job_->is_waiting()); EXPECT_TRUE(job_->is_delivering_network_response()); // When the request is restarted, the existing job is dropped so a // real network job gets created. We expect NULL here which will cause // the net library to create a real job. job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_FALSE(job_); // Simulate an http error of the real network job, but with custom // headers that override the fallback behavior. const char kOverrideHeaders[] = "HTTP/1.1 404 BOO HOO\0" "x-chromium-appcache-fallback-override: disallow-fallback\0" "\0"; net::HttpResponseInfo info; info.headers = new net::HttpResponseHeaders( std::string(kOverrideHeaders, arraysize(kOverrideHeaders))); request_->SimulateResponseInfo(info); job_ = handler_->MaybeLoadFallbackForResponse(request_.get()); EXPECT_FALSE(job_); TestFinished(); } // SubResource_Miss_WithNoCacheSelected ---------------------------------- void SubResource_Miss_WithNoCacheSelected() { request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::SUB_RESOURCE)); // We avoid creating handler when possible, sub-resource requests are not // subject to retrieval from an appcache when there's no associated cache. EXPECT_FALSE(handler_.get()); TestFinished(); } // SubResource_Miss_WithCacheSelected ---------------------------------- void SubResource_Miss_WithCacheSelected() { // A sub-resource load where the resource is not in an appcache, or // in a network or fallback namespace, should result in a failed request. host_->AssociateCache(MakeNewCache()); request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::SUB_RESOURCE)); EXPECT_TRUE(handler_.get()); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_TRUE(job_.get()); EXPECT_TRUE(job_->is_delivering_error_response()); AppCacheURLRequestJob* fallback_job; fallback_job = handler_->MaybeLoadFallbackForRedirect( request_.get(), GURL("http://blah/redirect")); EXPECT_FALSE(fallback_job); fallback_job = handler_->MaybeLoadFallbackForResponse(request_.get()); EXPECT_FALSE(fallback_job); TestFinished(); } // SubResource_Miss_WithWaitForCacheSelection ----------------------------- void SubResource_Miss_WithWaitForCacheSelection() { // Precondition, the host is waiting on cache selection. scoped_refptr cache(MakeNewCache()); host_->pending_selected_cache_id_ = cache->cache_id(); host_->set_preferred_manifest_url(cache->owning_group()->manifest_url()); request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::SUB_RESOURCE)); EXPECT_TRUE(handler_.get()); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_TRUE(job_.get()); EXPECT_TRUE(job_->is_waiting()); host_->FinishCacheSelection(cache, NULL); EXPECT_FALSE(job_->is_waiting()); EXPECT_TRUE(job_->is_delivering_error_response()); AppCacheURLRequestJob* fallback_job; fallback_job = handler_->MaybeLoadFallbackForRedirect( request_.get(), GURL("http://blah/redirect")); EXPECT_FALSE(fallback_job); fallback_job = handler_->MaybeLoadFallbackForResponse(request_.get()); EXPECT_FALSE(fallback_job); TestFinished(); } // SubResource_Hit ----------------------------- void SubResource_Hit() { host_->AssociateCache(MakeNewCache()); mock_storage()->SimulateFindSubResource( AppCacheEntry(AppCacheEntry::EXPLICIT, 1), AppCacheEntry(), false); request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::SUB_RESOURCE)); EXPECT_TRUE(handler_.get()); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_TRUE(job_.get()); EXPECT_TRUE(job_->is_delivering_appcache_response()); AppCacheURLRequestJob* fallback_job; fallback_job = handler_->MaybeLoadFallbackForRedirect( request_.get(), GURL("http://blah/redirect")); EXPECT_FALSE(fallback_job); fallback_job = handler_->MaybeLoadFallbackForResponse(request_.get()); EXPECT_FALSE(fallback_job); TestFinished(); } // SubResource_RedirectFallback ----------------------------- void SubResource_RedirectFallback() { // Redirects to resources in the a different origin are subject to // fallback namespaces. host_->AssociateCache(MakeNewCache()); mock_storage()->SimulateFindSubResource( AppCacheEntry(), AppCacheEntry(AppCacheEntry::EXPLICIT, 1), false); request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::SUB_RESOURCE)); EXPECT_TRUE(handler_.get()); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_FALSE(job_.get()); job_ = handler_->MaybeLoadFallbackForRedirect( request_.get(), GURL("http://not_blah/redirect")); EXPECT_TRUE(job_.get()); EXPECT_TRUE(job_->is_delivering_appcache_response()); AppCacheURLRequestJob* fallback_job; fallback_job = handler_->MaybeLoadFallbackForResponse(request_.get()); EXPECT_FALSE(fallback_job); TestFinished(); } // SubResource_NoRedirectFallback ----------------------------- void SubResource_NoRedirectFallback() { // Redirects to resources in the same-origin are not subject to // fallback namespaces. host_->AssociateCache(MakeNewCache()); mock_storage()->SimulateFindSubResource( AppCacheEntry(), AppCacheEntry(AppCacheEntry::EXPLICIT, 1), false); request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::SUB_RESOURCE)); EXPECT_TRUE(handler_.get()); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_FALSE(job_.get()); AppCacheURLRequestJob* fallback_job; fallback_job = handler_->MaybeLoadFallbackForRedirect( request_.get(), GURL("http://blah/redirect")); EXPECT_FALSE(fallback_job); request_->SimulateResponseCode(200); fallback_job = handler_->MaybeLoadFallbackForResponse(request_.get()); EXPECT_FALSE(fallback_job); TestFinished(); } // SubResource_Network ----------------------------- void SubResource_Network() { // A sub-resource load where the resource is in a network namespace, // should result in the system using a 'real' job to do the network // retrieval. host_->AssociateCache(MakeNewCache()); mock_storage()->SimulateFindSubResource( AppCacheEntry(), AppCacheEntry(), true); request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::SUB_RESOURCE)); EXPECT_TRUE(handler_.get()); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_FALSE(job_.get()); AppCacheURLRequestJob* fallback_job; fallback_job = handler_->MaybeLoadFallbackForRedirect( request_.get(), GURL("http://blah/redirect")); EXPECT_FALSE(fallback_job); fallback_job = handler_->MaybeLoadFallbackForResponse(request_.get()); EXPECT_FALSE(fallback_job); TestFinished(); } // DestroyedHost ----------------------------- void DestroyedHost() { host_->AssociateCache(MakeNewCache()); mock_storage()->SimulateFindSubResource( AppCacheEntry(AppCacheEntry::EXPLICIT, 1), AppCacheEntry(), false); request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::SUB_RESOURCE)); EXPECT_TRUE(handler_.get()); backend_impl_->UnregisterHost(1); host_ = NULL; EXPECT_FALSE(handler_->MaybeLoadResource(request_.get())); EXPECT_FALSE(handler_->MaybeLoadFallbackForRedirect( request_.get(), GURL("http://blah/redirect"))); EXPECT_FALSE(handler_->MaybeLoadFallbackForResponse(request_.get())); TestFinished(); } // DestroyedHostWithWaitingJob ----------------------------- void DestroyedHostWithWaitingJob() { // Precondition, the host is waiting on cache selection. host_->pending_selected_cache_id_ = 1; request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::SUB_RESOURCE)); EXPECT_TRUE(handler_.get()); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_TRUE(job_.get()); EXPECT_TRUE(job_->is_waiting()); backend_impl_->UnregisterHost(1); host_ = NULL; EXPECT_TRUE(job_->has_been_killed()); EXPECT_FALSE(handler_->MaybeLoadResource(request_.get())); EXPECT_FALSE(handler_->MaybeLoadFallbackForRedirect( request_.get(), GURL("http://blah/redirect"))); EXPECT_FALSE(handler_->MaybeLoadFallbackForResponse(request_.get())); TestFinished(); } // UnsupportedScheme ----------------------------- void UnsupportedScheme() { // Precondition, the host is waiting on cache selection. host_->pending_selected_cache_id_ = 1; request_.reset(new MockURLRequest(GURL("ftp://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::SUB_RESOURCE)); EXPECT_TRUE(handler_.get()); // we could redirect to http (conceivably) EXPECT_FALSE(handler_->MaybeLoadResource(request_.get())); EXPECT_FALSE(handler_->MaybeLoadFallbackForRedirect( request_.get(), GURL("ftp://blah/redirect"))); EXPECT_FALSE(handler_->MaybeLoadFallbackForResponse(request_.get())); TestFinished(); } // CanceledRequest ----------------------------- void CanceledRequest() { request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::MAIN_FRAME)); EXPECT_TRUE(handler_.get()); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_TRUE(job_.get()); EXPECT_TRUE(job_->is_waiting()); EXPECT_FALSE(job_->has_been_started()); mock_factory_job_ = job_.get(); request_->Start(); EXPECT_TRUE(job_->has_been_started()); request_->Cancel(); EXPECT_TRUE(job_->has_been_killed()); EXPECT_FALSE(handler_->MaybeLoadFallbackForResponse(request_.get())); TestFinished(); } // WorkerRequest ----------------------------- void WorkerRequest() { EXPECT_TRUE(AppCacheRequestHandler::IsMainResourceType( ResourceType::MAIN_FRAME)); EXPECT_TRUE(AppCacheRequestHandler::IsMainResourceType( ResourceType::SUB_FRAME)); EXPECT_TRUE(AppCacheRequestHandler::IsMainResourceType( ResourceType::SHARED_WORKER)); EXPECT_FALSE(AppCacheRequestHandler::IsMainResourceType( ResourceType::WORKER)); request_.reset(new MockURLRequest(GURL("http://blah/"))); const int kParentHostId = host_->host_id(); const int kWorkerHostId = 2; const int kAbandonedWorkerHostId = 3; const int kNonExsitingHostId = 700; backend_impl_->RegisterHost(kWorkerHostId); AppCacheHost* worker_host = backend_impl_->GetHost(kWorkerHostId); worker_host->SelectCacheForWorker(kParentHostId, kMockProcessId); handler_.reset(worker_host->CreateRequestHandler( request_.get(), ResourceType::SHARED_WORKER)); EXPECT_TRUE(handler_.get()); // Verify that the handler is associated with the parent host. EXPECT_EQ(host_, handler_->host_); // Create a new worker host, but associate it with a parent host that // does not exists to simulate the host having been torn down. backend_impl_->UnregisterHost(kWorkerHostId); backend_impl_->RegisterHost(kAbandonedWorkerHostId); worker_host = backend_impl_->GetHost(kAbandonedWorkerHostId); EXPECT_EQ(NULL, backend_impl_->GetHost(kNonExsitingHostId)); worker_host->SelectCacheForWorker(kNonExsitingHostId, kMockProcessId); handler_.reset(worker_host->CreateRequestHandler( request_.get(), ResourceType::SHARED_WORKER)); EXPECT_FALSE(handler_.get()); TestFinished(); } // MainResource_Blocked -------------------------------------------------- void MainResource_Blocked() { PushNextTask(NewRunnableMethod( this, &AppCacheRequestHandlerTest::Verify_MainResource_Blocked)); request_.reset(new MockURLRequest(GURL("http://blah/"))); handler_.reset(host_->CreateRequestHandler(request_.get(), ResourceType::MAIN_FRAME)); EXPECT_TRUE(handler_.get()); mock_policy_->can_load_return_value_ = false; mock_storage()->SimulateFindMainResource( AppCacheEntry(AppCacheEntry::EXPLICIT, 1), GURL(), AppCacheEntry(), 1, GURL("http://blah/manifest/")); job_ = handler_->MaybeLoadResource(request_.get()); EXPECT_TRUE(job_.get()); EXPECT_TRUE(job_->is_waiting()); // We have to wait for completion of storage->FindResponseForMainRequest. ScheduleNextTask(); } void Verify_MainResource_Blocked() { EXPECT_FALSE(job_->is_waiting()); EXPECT_FALSE(job_->is_delivering_appcache_response()); EXPECT_EQ(0, handler_->found_cache_id_); EXPECT_TRUE(handler_->found_manifest_url_.is_empty()); EXPECT_TRUE(host_->preferred_manifest_url().is_empty()); EXPECT_TRUE(host_->main_resource_blocked_); EXPECT_TRUE(host_->blocked_manifest_url_ == GURL("http://blah/manifest/")); TestFinished(); } // Test case helpers -------------------------------------------------- AppCache* MakeNewCache() { AppCache* cache = new AppCache( mock_service_.get(), mock_storage()->NewCacheId()); cache->set_complete(true); AppCacheGroup* group = new AppCacheGroup( mock_service_.get(), GURL("http://blah/manifest"), mock_storage()->NewGroupId()); group->AddCache(cache); return cache; } MockAppCacheStorage* mock_storage() { return reinterpret_cast(mock_service_->storage()); } // Data members -------------------------------------------------- scoped_ptr test_finished_event_; std::stack task_stack_; scoped_ptr mock_service_; scoped_ptr backend_impl_; scoped_ptr mock_frontend_; scoped_ptr mock_policy_; AppCacheHost* host_; scoped_ptr request_; scoped_ptr handler_; scoped_refptr job_; net::URLRequest::ProtocolFactory* orig_http_factory_; static scoped_ptr io_thread_; static net::URLRequestJob* mock_factory_job_; }; // static scoped_ptr AppCacheRequestHandlerTest::io_thread_; net::URLRequestJob* AppCacheRequestHandlerTest::mock_factory_job_ = NULL; TEST_F(AppCacheRequestHandlerTest, MainResource_Miss) { RunTestOnIOThread(&AppCacheRequestHandlerTest::MainResource_Miss); } TEST_F(AppCacheRequestHandlerTest, MainResource_Hit) { RunTestOnIOThread(&AppCacheRequestHandlerTest::MainResource_Hit); } TEST_F(AppCacheRequestHandlerTest, MainResource_Fallback) { RunTestOnIOThread(&AppCacheRequestHandlerTest::MainResource_Fallback); } TEST_F(AppCacheRequestHandlerTest, MainResource_FallbackOverride) { RunTestOnIOThread( &AppCacheRequestHandlerTest::MainResource_FallbackOverride); } TEST_F(AppCacheRequestHandlerTest, SubResource_Miss_WithNoCacheSelected) { RunTestOnIOThread( &AppCacheRequestHandlerTest::SubResource_Miss_WithNoCacheSelected); } TEST_F(AppCacheRequestHandlerTest, SubResource_Miss_WithCacheSelected) { RunTestOnIOThread( &AppCacheRequestHandlerTest::SubResource_Miss_WithCacheSelected); } TEST_F(AppCacheRequestHandlerTest, SubResource_Miss_WithWaitForCacheSelection) { RunTestOnIOThread( &AppCacheRequestHandlerTest::SubResource_Miss_WithWaitForCacheSelection); } TEST_F(AppCacheRequestHandlerTest, SubResource_Hit) { RunTestOnIOThread(&AppCacheRequestHandlerTest::SubResource_Hit); } TEST_F(AppCacheRequestHandlerTest, SubResource_RedirectFallback) { RunTestOnIOThread(&AppCacheRequestHandlerTest::SubResource_RedirectFallback); } TEST_F(AppCacheRequestHandlerTest, SubResource_NoRedirectFallback) { RunTestOnIOThread( &AppCacheRequestHandlerTest::SubResource_NoRedirectFallback); } TEST_F(AppCacheRequestHandlerTest, SubResource_Network) { RunTestOnIOThread(&AppCacheRequestHandlerTest::SubResource_Network); } TEST_F(AppCacheRequestHandlerTest, DestroyedHost) { RunTestOnIOThread(&AppCacheRequestHandlerTest::DestroyedHost); } TEST_F(AppCacheRequestHandlerTest, DestroyedHostWithWaitingJob) { RunTestOnIOThread(&AppCacheRequestHandlerTest::DestroyedHostWithWaitingJob); } TEST_F(AppCacheRequestHandlerTest, UnsupportedScheme) { RunTestOnIOThread(&AppCacheRequestHandlerTest::UnsupportedScheme); } TEST_F(AppCacheRequestHandlerTest, CanceledRequest) { RunTestOnIOThread(&AppCacheRequestHandlerTest::CanceledRequest); } TEST_F(AppCacheRequestHandlerTest, WorkerRequest) { RunTestOnIOThread(&AppCacheRequestHandlerTest::WorkerRequest); } TEST_F(AppCacheRequestHandlerTest, MainResource_Blocked) { RunTestOnIOThread(&AppCacheRequestHandlerTest::MainResource_Blocked); } } // namespace appcache // AppCacheRequestHandlerTest is expected to always live longer than the // runnable methods. This lets us call NewRunnableMethod on its instances. DISABLE_RUNNABLE_METHOD_REFCOUNT(appcache::AppCacheRequestHandlerTest);