| // 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 "base/bind.h" | 
 | #include "base/files/file_util.h" | 
 | #include "base/files/scoped_temp_dir.h" | 
 | #include "base/location.h" | 
 | #include "base/memory/ref_counted.h" | 
 | #include "base/run_loop.h" | 
 | #include "base/single_thread_task_runner.h" | 
 | #include "content/browser/appcache/appcache_database.h" | 
 | #include "content/browser/appcache/appcache_storage_impl.h" | 
 | #include "content/browser/appcache/chrome_appcache_service.h" | 
 | #include "content/browser/browser_thread_impl.h" | 
 | #include "content/public/browser/resource_context.h" | 
 | #include "content/public/test/mock_special_storage_policy.h" | 
 | #include "content/public/test/test_browser_context.h" | 
 | #include "content/test/appcache_test_helper.h" | 
 | #include "net/url_request/url_request_context_getter.h" | 
 | #include "testing/gtest/include/gtest/gtest.h" | 
 |  | 
 | #include <set> | 
 |  | 
 | namespace content { | 
 | namespace { | 
 | const base::FilePath::CharType kTestingAppCacheDirname[] = | 
 |     FILE_PATH_LITERAL("Application Cache"); | 
 |  | 
 | // Examples of a protected and an unprotected origin, to be used througout the | 
 | // test. | 
 | const char kProtectedManifest[] = "http://www.protected.com/cache.manifest"; | 
 | const char kNormalManifest[] = "http://www.normal.com/cache.manifest"; | 
 | const char kSessionOnlyManifest[] = "http://www.sessiononly.com/cache.manifest"; | 
 |  | 
 | class MockURLRequestContextGetter : public net::URLRequestContextGetter { | 
 |  public: | 
 |   MockURLRequestContextGetter(net::URLRequestContext* context, | 
 |                               base::SingleThreadTaskRunner* task_runner) | 
 |       : context_(context), task_runner_(task_runner) {} | 
 |  | 
 |   net::URLRequestContext* GetURLRequestContext() override { return context_; } | 
 |  | 
 |   scoped_refptr<base::SingleThreadTaskRunner> GetNetworkTaskRunner() | 
 |       const override { | 
 |     return task_runner_; | 
 |   } | 
 |  | 
 |  protected: | 
 |   ~MockURLRequestContextGetter() override {} | 
 |  | 
 |  private: | 
 |   net::URLRequestContext* context_; | 
 |   scoped_refptr<base::SingleThreadTaskRunner> task_runner_; | 
 | }; | 
 |  | 
 | }  // namespace | 
 |  | 
 | class ChromeAppCacheServiceTest : public testing::Test { | 
 |  public: | 
 |   ChromeAppCacheServiceTest() | 
 |       : kProtectedManifestURL(kProtectedManifest), | 
 |         kNormalManifestURL(kNormalManifest), | 
 |         kSessionOnlyManifestURL(kSessionOnlyManifest), | 
 |         file_thread_(BrowserThread::FILE, &message_loop_), | 
 |         file_user_blocking_thread_(BrowserThread::FILE_USER_BLOCKING, | 
 |                                    &message_loop_), | 
 |         cache_thread_(BrowserThread::CACHE, &message_loop_), | 
 |         io_thread_(BrowserThread::IO, &message_loop_) {} | 
 |  | 
 |  protected: | 
 |   scoped_refptr<ChromeAppCacheService> CreateAppCacheServiceImpl( | 
 |       const base::FilePath& appcache_path, | 
 |       bool init_storage); | 
 |   void InsertDataIntoAppCache(ChromeAppCacheService* appcache_service); | 
 |  | 
 |   base::MessageLoopForIO message_loop_; | 
 |   base::ScopedTempDir temp_dir_; | 
 |   const GURL kProtectedManifestURL; | 
 |   const GURL kNormalManifestURL; | 
 |   const GURL kSessionOnlyManifestURL; | 
 |  | 
 |  private: | 
 |   BrowserThreadImpl file_thread_; | 
 |   BrowserThreadImpl file_user_blocking_thread_; | 
 |   BrowserThreadImpl cache_thread_; | 
 |   BrowserThreadImpl io_thread_; | 
 |   TestBrowserContext browser_context_; | 
 | }; | 
 |  | 
 | scoped_refptr<ChromeAppCacheService> | 
 | ChromeAppCacheServiceTest::CreateAppCacheServiceImpl( | 
 |     const base::FilePath& appcache_path, | 
 |     bool init_storage) { | 
 |   scoped_refptr<ChromeAppCacheService> appcache_service = | 
 |       new ChromeAppCacheService(NULL); | 
 |   scoped_refptr<MockSpecialStoragePolicy> mock_policy = | 
 |       new MockSpecialStoragePolicy; | 
 |   mock_policy->AddProtected(kProtectedManifestURL.GetOrigin()); | 
 |   mock_policy->AddSessionOnly(kSessionOnlyManifestURL.GetOrigin()); | 
 |   scoped_refptr<MockURLRequestContextGetter> mock_request_context_getter = | 
 |       new MockURLRequestContextGetter( | 
 |           browser_context_.GetResourceContext()->GetRequestContext(), | 
 |           message_loop_.task_runner().get()); | 
 |   BrowserThread::PostTask( | 
 |       BrowserThread::IO, FROM_HERE, | 
 |       base::Bind(&ChromeAppCacheService::InitializeOnIOThread, | 
 |                  appcache_service, appcache_path, | 
 |                  browser_context_.GetResourceContext(), | 
 |                  base::RetainedRef(mock_request_context_getter), mock_policy)); | 
 |   // Steps needed to initialize the storage of AppCache data. | 
 |   base::RunLoop().RunUntilIdle(); | 
 |   if (init_storage) { | 
 |     AppCacheStorageImpl* storage = | 
 |         static_cast<AppCacheStorageImpl*>( | 
 |             appcache_service->storage()); | 
 |     storage->database_->db_connection(); | 
 |     storage->disk_cache(); | 
 |     base::RunLoop().RunUntilIdle(); | 
 |   } | 
 |   return appcache_service; | 
 | } | 
 |  | 
 | void ChromeAppCacheServiceTest::InsertDataIntoAppCache( | 
 |     ChromeAppCacheService* appcache_service) { | 
 |   AppCacheTestHelper appcache_helper; | 
 |   appcache_helper.AddGroupAndCache(appcache_service, kNormalManifestURL); | 
 |   appcache_helper.AddGroupAndCache(appcache_service, kProtectedManifestURL); | 
 |   appcache_helper.AddGroupAndCache(appcache_service, kSessionOnlyManifestURL); | 
 |  | 
 |   // Verify that adding the data succeeded | 
 |   std::set<GURL> origins; | 
 |   appcache_helper.GetOriginsWithCaches(appcache_service, &origins); | 
 |   ASSERT_EQ(3UL, origins.size()); | 
 |   ASSERT_TRUE(origins.find(kProtectedManifestURL.GetOrigin()) != origins.end()); | 
 |   ASSERT_TRUE(origins.find(kNormalManifestURL.GetOrigin()) != origins.end()); | 
 |   ASSERT_TRUE(origins.find(kSessionOnlyManifestURL.GetOrigin()) != | 
 |               origins.end()); | 
 | } | 
 |  | 
 | TEST_F(ChromeAppCacheServiceTest, KeepOnDestruction) { | 
 |   ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); | 
 |   base::FilePath appcache_path = | 
 |       temp_dir_.GetPath().Append(kTestingAppCacheDirname); | 
 |  | 
 |   // Create a ChromeAppCacheService and insert data into it | 
 |   scoped_refptr<ChromeAppCacheService> appcache_service = | 
 |       CreateAppCacheServiceImpl(appcache_path, true); | 
 |   ASSERT_TRUE(base::PathExists(appcache_path)); | 
 |   ASSERT_TRUE(base::PathExists(appcache_path.AppendASCII("Index"))); | 
 |   InsertDataIntoAppCache(appcache_service.get()); | 
 |  | 
 |   // Test: delete the ChromeAppCacheService | 
 |   appcache_service = NULL; | 
 |   base::RunLoop().RunUntilIdle(); | 
 |  | 
 |   // Recreate the appcache (for reading the data back) | 
 |   appcache_service = CreateAppCacheServiceImpl(appcache_path, false); | 
 |  | 
 |   // The directory is still there | 
 |   ASSERT_TRUE(base::PathExists(appcache_path)); | 
 |  | 
 |   // The appcache data is also there, except the session-only origin. | 
 |   AppCacheTestHelper appcache_helper; | 
 |   std::set<GURL> origins; | 
 |   appcache_helper.GetOriginsWithCaches(appcache_service.get(), &origins); | 
 |   EXPECT_EQ(2UL, origins.size()); | 
 |   EXPECT_TRUE(origins.find(kProtectedManifestURL.GetOrigin()) != origins.end()); | 
 |   EXPECT_TRUE(origins.find(kNormalManifestURL.GetOrigin()) != origins.end()); | 
 |   EXPECT_TRUE(origins.find(kSessionOnlyManifestURL.GetOrigin()) == | 
 |               origins.end()); | 
 |  | 
 |   // Delete and let cleanup tasks run prior to returning. | 
 |   appcache_service = NULL; | 
 |   base::RunLoop().RunUntilIdle(); | 
 | } | 
 |  | 
 | TEST_F(ChromeAppCacheServiceTest, SaveSessionState) { | 
 |   ASSERT_TRUE(temp_dir_.CreateUniqueTempDir()); | 
 |   base::FilePath appcache_path = | 
 |       temp_dir_.GetPath().Append(kTestingAppCacheDirname); | 
 |  | 
 |   // Create a ChromeAppCacheService and insert data into it | 
 |   scoped_refptr<ChromeAppCacheService> appcache_service = | 
 |       CreateAppCacheServiceImpl(appcache_path, true); | 
 |   ASSERT_TRUE(base::PathExists(appcache_path)); | 
 |   ASSERT_TRUE(base::PathExists(appcache_path.AppendASCII("Index"))); | 
 |   InsertDataIntoAppCache(appcache_service.get()); | 
 |  | 
 |   // Save session state. This should bypass the destruction-time deletion. | 
 |   appcache_service->set_force_keep_session_state(); | 
 |  | 
 |   // Test: delete the ChromeAppCacheService | 
 |   appcache_service = NULL; | 
 |   base::RunLoop().RunUntilIdle(); | 
 |  | 
 |   // Recreate the appcache (for reading the data back) | 
 |   appcache_service = CreateAppCacheServiceImpl(appcache_path, false); | 
 |  | 
 |   // The directory is still there | 
 |   ASSERT_TRUE(base::PathExists(appcache_path)); | 
 |  | 
 |   // No appcache data was deleted. | 
 |   AppCacheTestHelper appcache_helper; | 
 |   std::set<GURL> origins; | 
 |   appcache_helper.GetOriginsWithCaches(appcache_service.get(), &origins); | 
 |   EXPECT_EQ(3UL, origins.size()); | 
 |   EXPECT_TRUE(origins.find(kProtectedManifestURL.GetOrigin()) != origins.end()); | 
 |   EXPECT_TRUE(origins.find(kNormalManifestURL.GetOrigin()) != origins.end()); | 
 |   EXPECT_TRUE(origins.find(kSessionOnlyManifestURL.GetOrigin()) != | 
 |               origins.end()); | 
 |  | 
 |   // Delete and let cleanup tasks run prior to returning. | 
 |   appcache_service = NULL; | 
 |   base::RunLoop().RunUntilIdle(); | 
 | } | 
 |  | 
 | }  // namespace content |