// Copyright (c) 2009 The Chromium Authos. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "testing/gtest/include/gtest/gtest.h" #include "webkit/appcache/appcache.h" #include "webkit/appcache/appcache_group.h" #include "webkit/appcache/appcache_host.h" #include "webkit/appcache/mock_appcache_service.h" #include "webkit/appcache/appcache_update_job.h" namespace { class TestAppCacheFrontend : public appcache::AppCacheFrontend { public: TestAppCacheFrontend() : last_host_id_(-1), last_cache_id_(-1), last_status_(appcache::OBSOLETE) { } virtual void OnCacheSelected(int host_id, int64 cache_id , appcache::Status status) { last_host_id_ = host_id; last_cache_id_ = cache_id; last_status_ = status; } virtual void OnStatusChanged(const std::vector& host_ids, appcache::Status status) { } virtual void OnEventRaised(const std::vector& host_ids, appcache::EventID event_id) { } int last_host_id_; int64 last_cache_id_; appcache::Status last_status_; }; } // namespace anon namespace appcache { class TestUpdateObserver : public AppCacheGroup::UpdateObserver { public: TestUpdateObserver() : update_completed_(false), group_has_cache_(false) { } virtual void OnUpdateComplete(AppCacheGroup* group) { update_completed_ = true; group_has_cache_ = group->HasCache(); } bool update_completed_; bool group_has_cache_; }; class AppCacheGroupTest : public testing::Test { }; TEST(AppCacheGroupTest, AddRemoveCache) { MockAppCacheService service; scoped_refptr group = new AppCacheGroup(&service, GURL::EmptyGURL()); base::TimeTicks ticks = base::TimeTicks::Now(); scoped_refptr cache1 = new AppCache(&service, 111); cache1->set_complete(true); cache1->set_update_time(ticks); group->AddCache(cache1); EXPECT_EQ(cache1, group->newest_complete_cache()); // Adding older cache does not change newest complete cache. scoped_refptr cache2 = new AppCache(&service, 222); cache2->set_complete(true); cache2->set_update_time(ticks - base::TimeDelta::FromDays(1)); group->AddCache(cache2); EXPECT_EQ(cache1, group->newest_complete_cache()); // Adding newer cache does change newest complete cache. scoped_refptr cache3 = new AppCache(&service, 333); cache3->set_complete(true); cache3->set_update_time(ticks + base::TimeDelta::FromDays(1)); group->AddCache(cache3); EXPECT_EQ(cache3, group->newest_complete_cache()); // Adding cache with same update time uses one with larger ID. scoped_refptr cache4 = new AppCache(&service, 444); cache4->set_complete(true); cache4->set_update_time(ticks + base::TimeDelta::FromDays(1)); // same as 3 group->AddCache(cache4); EXPECT_EQ(cache4, group->newest_complete_cache()); scoped_refptr cache5 = new AppCache(&service, 55); // smaller id cache5->set_complete(true); cache5->set_update_time(ticks + base::TimeDelta::FromDays(1)); // same as 4 group->AddCache(cache5); EXPECT_EQ(cache4, group->newest_complete_cache()); // no change // Old caches can always be removed. group->RemoveCache(cache1); EXPECT_FALSE(cache1->owning_group()); EXPECT_EQ(cache4, group->newest_complete_cache()); // newest unchanged // Remove rest of caches. group->RemoveCache(cache2); EXPECT_FALSE(cache2->owning_group()); EXPECT_EQ(cache4, group->newest_complete_cache()); // newest unchanged group->RemoveCache(cache3); EXPECT_FALSE(cache3->owning_group()); EXPECT_EQ(cache4, group->newest_complete_cache()); // newest unchanged group->RemoveCache(cache5); EXPECT_FALSE(cache5->owning_group()); EXPECT_EQ(cache4, group->newest_complete_cache()); // newest unchanged group->RemoveCache(cache4); // newest removed EXPECT_FALSE(cache4->owning_group()); EXPECT_FALSE(group->newest_complete_cache()); // no more newest cache // Can remove newest cache if there are older caches. group->AddCache(cache1); EXPECT_EQ(cache1, group->newest_complete_cache()); group->AddCache(cache4); EXPECT_EQ(cache4, group->newest_complete_cache()); group->RemoveCache(cache4); // remove newest EXPECT_FALSE(cache4->owning_group()); EXPECT_FALSE(group->newest_complete_cache()); // newest removed } TEST(AppCacheGroupTest, CleanupUnusedGroup) { MockAppCacheService service; TestAppCacheFrontend frontend; AppCacheGroup* group = new AppCacheGroup(&service, GURL::EmptyGURL()); AppCacheHost host1(1, &frontend, &service); AppCacheHost host2(2, &frontend, &service); base::TimeTicks ticks = base::TimeTicks::Now(); AppCache* cache1 = new AppCache(&service, 111); cache1->set_complete(true); cache1->set_update_time(ticks); group->AddCache(cache1); EXPECT_EQ(cache1, group->newest_complete_cache()); host1.AssociateCache(cache1); EXPECT_EQ(frontend.last_host_id_, host1.host_id()); EXPECT_EQ(frontend.last_cache_id_, cache1->cache_id()); EXPECT_EQ(frontend.last_status_, appcache::IDLE); host2.AssociateCache(cache1); EXPECT_EQ(frontend.last_host_id_, host2.host_id()); EXPECT_EQ(frontend.last_cache_id_, cache1->cache_id()); EXPECT_EQ(frontend.last_status_, appcache::IDLE); AppCache* cache2 = new AppCache(&service, 222); cache2->set_complete(true); cache2->set_update_time(ticks + base::TimeDelta::FromDays(1)); group->AddCache(cache2); EXPECT_EQ(cache2, group->newest_complete_cache()); // Unassociate all hosts from older cache. host1.AssociateCache(NULL); host2.AssociateCache(NULL); EXPECT_EQ(frontend.last_host_id_, host2.host_id()); EXPECT_EQ(frontend.last_cache_id_, appcache::kNoCacheId); EXPECT_EQ(frontend.last_status_, appcache::UNCACHED); } TEST(AppCacheGroupTest, StartUpdate) { MockAppCacheService service; scoped_refptr group = new AppCacheGroup(&service, GURL("http://foo.com")); // Set state to checking to prevent update job from executing fetches. group->update_status_ = AppCacheGroup::CHECKING; group->StartUpdate(); AppCacheUpdateJob* update = group->update_job_; EXPECT_TRUE(update != NULL); // Start another update, check that same update job is in use. group->StartUpdateWithHost(NULL); EXPECT_EQ(update, group->update_job_); // Deleting the update should restore the group to IDLE. delete update; EXPECT_TRUE(group->update_job_ == NULL); EXPECT_EQ(AppCacheGroup::IDLE, group->update_status()); } TEST(AppCacheGroupTest, CancelUpdate) { MockAppCacheService service; scoped_refptr group = new AppCacheGroup(&service, GURL("http://foo.com")); // Set state to checking to prevent update job from executing fetches. group->update_status_ = AppCacheGroup::CHECKING; group->StartUpdate(); AppCacheUpdateJob* update = group->update_job_; EXPECT_TRUE(update != NULL); // Deleting the group should cancel the update. TestUpdateObserver observer; group->AddUpdateObserver(&observer); group = NULL; // causes group to be deleted EXPECT_TRUE(observer.update_completed_); EXPECT_FALSE(observer.group_has_cache_); } } // namespace appcache