// Copyright 2013 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 "components/precache/content/precache_manager.h" #include <list> #include <map> #include <set> #include <string> #include "base/basictypes.h" #include "base/bind.h" #include "base/callback.h" #include "base/command_line.h" #include "base/compiler_specific.h" #include "base/message_loop/message_loop.h" #include "base/metrics/histogram.h" #include "base/metrics/histogram_samples.h" #include "base/metrics/statistics_recorder.h" #include "components/precache/core/precache_switches.h" #include "components/precache/core/url_list_provider.h" #include "content/public/test/test_browser_context.h" #include "content/public/test/test_browser_thread_bundle.h" #include "net/http/http_status_code.h" #include "net/url_request/test_url_fetcher_factory.h" #include "net/url_request/url_request_status.h" #include "net/url_request/url_request_test_util.h" #include "testing/gtest/include/gtest/gtest.h" #include "url/gurl.h" namespace precache { namespace { // A map of histogram names to the total sample counts. typedef std::map<std::string, base::HistogramBase::Count> HistogramCountMap; const char kConfigURL[] = "http://config-url.com"; const char kManifestURLPrefix[] = "http://manifest-url-prefix.com/"; base::HistogramBase::Count GetHistogramTotalCount(const char* histogram_name) { base::HistogramBase* histogram = base::StatisticsRecorder::FindHistogram(histogram_name); return histogram ? histogram->SnapshotSamples()->TotalCount() : 0; } HistogramCountMap GetHistogramCountMap() { // Note that the PrecacheManager tests don't care about the ".Cellular" // histograms. const char* kHistogramNames[] = {"Precache.DownloadedPrecacheMotivated", "Precache.DownloadedNonPrecache", "Precache.Saved"}; HistogramCountMap histogram_count_map; for (size_t i = 0; i < arraysize(kHistogramNames); ++i) { histogram_count_map[kHistogramNames[i]] = GetHistogramTotalCount(kHistogramNames[i]); } return histogram_count_map; } class TestURLFetcherCallback { public: scoped_ptr<net::FakeURLFetcher> CreateURLFetcher( const GURL& url, net::URLFetcherDelegate* delegate, const std::string& response_data, net::HttpStatusCode response_code, net::URLRequestStatus::Status status) { scoped_ptr<net::FakeURLFetcher> fetcher(new net::FakeURLFetcher( url, delegate, response_data, response_code, status)); requested_urls_.insert(url); return fetcher.Pass(); } const std::multiset<GURL>& requested_urls() const { return requested_urls_; } private: // Multiset with one entry for each URL requested. std::multiset<GURL> requested_urls_; }; class FakeURLListProvider : public URLListProvider { public: FakeURLListProvider(const std::list<GURL>& urls, bool run_immediately) : urls_(urls), run_immediately_(run_immediately), was_get_urls_called_(false) {} virtual void GetURLs(const GetURLsCallback& callback) OVERRIDE { was_get_urls_called_ = true; if (run_immediately_) { callback.Run(urls_); } else { // Post the callback to be run later in the message loop. base::MessageLoop::current()->PostTask(FROM_HERE, base::Bind(callback, urls_)); } } bool was_get_urls_called() const { return was_get_urls_called_; } private: const std::list<GURL> urls_; const bool run_immediately_; bool was_get_urls_called_; }; class TestPrecacheCompletionCallback { public: TestPrecacheCompletionCallback() : was_on_done_called_(false) {} void OnDone() { was_on_done_called_ = true; } PrecacheManager::PrecacheCompletionCallback GetCallback() { return base::Bind(&TestPrecacheCompletionCallback::OnDone, base::Unretained(this)); } bool was_on_done_called() const { return was_on_done_called_; } private: bool was_on_done_called_; }; class PrecacheManagerTest : public testing::Test { public: PrecacheManagerTest() : precache_manager_(&browser_context_), factory_(NULL, base::Bind(&TestURLFetcherCallback::CreateURLFetcher, base::Unretained(&url_callback_))) {} protected: virtual void SetUp() OVERRIDE { base::StatisticsRecorder::Initialize(); CommandLine::ForCurrentProcess()->AppendSwitchASCII( switches::kPrecacheConfigSettingsURL, kConfigURL); CommandLine::ForCurrentProcess()->AppendSwitchASCII( switches::kPrecacheManifestURLPrefix, kManifestURLPrefix); // Make the fetch of the precache configuration settings fail. Precaching // should still complete normally in this case. factory_.SetFakeResponse(GURL(kConfigURL), "", net::HTTP_INTERNAL_SERVER_ERROR, net::URLRequestStatus::FAILED); } content::TestBrowserThreadBundle test_browser_thread_bundle_; content::TestBrowserContext browser_context_; PrecacheManager precache_manager_; TestURLFetcherCallback url_callback_; net::FakeURLFetcherFactory factory_; TestPrecacheCompletionCallback precache_callback_; }; TEST_F(PrecacheManagerTest, StartAndFinishPrecaching) { EXPECT_FALSE(precache_manager_.IsPrecaching()); FakeURLListProvider url_list_provider( std::list<GURL>(1, GURL("http://starting-url.com")), false); precache_manager_.StartPrecaching(precache_callback_.GetCallback(), &url_list_provider); EXPECT_TRUE(precache_manager_.IsPrecaching()); base::MessageLoop::current()->RunUntilIdle(); EXPECT_FALSE(precache_manager_.IsPrecaching()); EXPECT_TRUE(url_list_provider.was_get_urls_called()); EXPECT_TRUE(precache_callback_.was_on_done_called()); std::multiset<GURL> expected_requested_urls; expected_requested_urls.insert(GURL(kConfigURL)); EXPECT_EQ(expected_requested_urls, url_callback_.requested_urls()); } TEST_F(PrecacheManagerTest, StartAndCancelPrecachingBeforeURLsReceived) { EXPECT_FALSE(precache_manager_.IsPrecaching()); FakeURLListProvider url_list_provider( std::list<GURL>(1, GURL("http://starting-url.com")), false); precache_manager_.StartPrecaching(precache_callback_.GetCallback(), &url_list_provider); EXPECT_TRUE(precache_manager_.IsPrecaching()); precache_manager_.CancelPrecaching(); EXPECT_FALSE(precache_manager_.IsPrecaching()); base::MessageLoop::current()->RunUntilIdle(); EXPECT_FALSE(precache_manager_.IsPrecaching()); EXPECT_TRUE(url_list_provider.was_get_urls_called()); EXPECT_FALSE(precache_callback_.was_on_done_called()); EXPECT_TRUE(url_callback_.requested_urls().empty()); } TEST_F(PrecacheManagerTest, StartAndCancelPrecachingAfterURLsReceived) { EXPECT_FALSE(precache_manager_.IsPrecaching()); FakeURLListProvider url_list_provider( std::list<GURL>(1, GURL("http://starting-url.com")), true); precache_manager_.StartPrecaching(precache_callback_.GetCallback(), &url_list_provider); // Since the |url_list_provider| ran the callback immediately, Start() has // been called on the PrecacheFetcher, and the precache config settings have // been requested. The response has not yet been received though, so // precaching is still in progress. EXPECT_TRUE(precache_manager_.IsPrecaching()); precache_manager_.CancelPrecaching(); EXPECT_FALSE(precache_manager_.IsPrecaching()); base::MessageLoop::current()->RunUntilIdle(); EXPECT_FALSE(precache_manager_.IsPrecaching()); EXPECT_TRUE(url_list_provider.was_get_urls_called()); EXPECT_FALSE(precache_callback_.was_on_done_called()); // Even though the response for the precache config settings should not have // been received, the request should still have been made. std::multiset<GURL> expected_requested_urls; expected_requested_urls.insert(GURL(kConfigURL)); EXPECT_EQ(expected_requested_urls, url_callback_.requested_urls()); } TEST_F(PrecacheManagerTest, RecordStatsForFetchWithIrrelevantFetches) { HistogramCountMap expected_histogram_count_map = GetHistogramCountMap(); // Fetches with size 0 should be ignored. precache_manager_.RecordStatsForFetch(GURL("http://url.com"), base::Time(), 0, false); base::MessageLoop::current()->RunUntilIdle(); EXPECT_EQ(expected_histogram_count_map, GetHistogramCountMap()); // Fetches for URLs with schemes other than HTTP or HTTPS should be ignored. precache_manager_.RecordStatsForFetch(GURL("ftp://ftp.com"), base::Time(), 1000, false); base::MessageLoop::current()->RunUntilIdle(); EXPECT_EQ(expected_histogram_count_map, GetHistogramCountMap()); // Fetches for empty URLs should be ignored. precache_manager_.RecordStatsForFetch(GURL(), base::Time(), 1000, false); base::MessageLoop::current()->RunUntilIdle(); EXPECT_EQ(expected_histogram_count_map, GetHistogramCountMap()); } TEST_F(PrecacheManagerTest, RecordStatsForFetchDuringPrecaching) { HistogramCountMap expected_histogram_count_map = GetHistogramCountMap(); FakeURLListProvider url_list_provider(std::list<GURL>(), false); precache_manager_.StartPrecaching(precache_callback_.GetCallback(), &url_list_provider); EXPECT_TRUE(precache_manager_.IsPrecaching()); precache_manager_.RecordStatsForFetch(GURL("http://url.com"), base::Time(), 1000, false); precache_manager_.CancelPrecaching(); base::MessageLoop::current()->RunUntilIdle(); expected_histogram_count_map["Precache.DownloadedPrecacheMotivated"]++; EXPECT_EQ(expected_histogram_count_map, GetHistogramCountMap()); } TEST_F(PrecacheManagerTest, RecordStatsForFetchHTTP) { HistogramCountMap expected_histogram_count_map = GetHistogramCountMap(); precache_manager_.RecordStatsForFetch(GURL("http://http-url.com"), base::Time(), 1000, false); base::MessageLoop::current()->RunUntilIdle(); expected_histogram_count_map["Precache.DownloadedNonPrecache"]++; EXPECT_EQ(expected_histogram_count_map, GetHistogramCountMap()); } TEST_F(PrecacheManagerTest, RecordStatsForFetchHTTPS) { HistogramCountMap expected_histogram_count_map = GetHistogramCountMap(); precache_manager_.RecordStatsForFetch(GURL("https://https-url.com"), base::Time(), 1000, false); base::MessageLoop::current()->RunUntilIdle(); expected_histogram_count_map["Precache.DownloadedNonPrecache"]++; EXPECT_EQ(expected_histogram_count_map, GetHistogramCountMap()); } TEST_F(PrecacheManagerTest, DeleteExpiredPrecacheHistory) { // This test has to use Time::Now() because StartPrecaching uses Time::Now(). const base::Time kCurrentTime = base::Time::Now(); HistogramCountMap expected_histogram_count_map = GetHistogramCountMap(); FakeURLListProvider url_list_provider(std::list<GURL>(), false); precache_manager_.StartPrecaching(precache_callback_.GetCallback(), &url_list_provider); EXPECT_TRUE(precache_manager_.IsPrecaching()); // Precache a bunch of URLs, with different fetch times. precache_manager_.RecordStatsForFetch( GURL("http://old-fetch.com"), kCurrentTime - base::TimeDelta::FromDays(61), 1000, false); precache_manager_.RecordStatsForFetch( GURL("http://recent-fetch.com"), kCurrentTime - base::TimeDelta::FromDays(59), 1000, false); precache_manager_.RecordStatsForFetch( GURL("http://yesterday-fetch.com"), kCurrentTime - base::TimeDelta::FromDays(1), 1000, false); expected_histogram_count_map["Precache.DownloadedPrecacheMotivated"] += 3; precache_manager_.CancelPrecaching(); base::MessageLoop::current()->RunUntilIdle(); EXPECT_EQ(expected_histogram_count_map, GetHistogramCountMap()); // The expired precache will be deleted during precaching this time. precache_manager_.StartPrecaching(precache_callback_.GetCallback(), &url_list_provider); EXPECT_TRUE(precache_manager_.IsPrecaching()); precache_manager_.CancelPrecaching(); base::MessageLoop::current()->RunUntilIdle(); EXPECT_FALSE(precache_manager_.IsPrecaching()); // A fetch for the same URL as the expired precache was served from the cache, // but it isn't reported as saved bytes because it had expired in the precache // history. precache_manager_.RecordStatsForFetch( GURL("http://old-fetch.com"), kCurrentTime, 1000, true); base::MessageLoop::current()->RunUntilIdle(); EXPECT_EQ(expected_histogram_count_map, GetHistogramCountMap()); // The other precaches should not have expired, so the following fetches from // the cache should count as saved bytes. precache_manager_.RecordStatsForFetch( GURL("http://recent-fetch.com"), kCurrentTime, 1000, true); precache_manager_.RecordStatsForFetch( GURL("http://yesterday-fetch.com"), kCurrentTime, 1000, true); expected_histogram_count_map["Precache.Saved"] += 2; base::MessageLoop::current()->RunUntilIdle(); EXPECT_EQ(expected_histogram_count_map, GetHistogramCountMap()); } } // namespace } // namespace precache