summaryrefslogtreecommitdiffstats
path: root/webkit
diff options
context:
space:
mode:
Diffstat (limited to 'webkit')
-rw-r--r--webkit/appcache/appcache_group.cc117
-rw-r--r--webkit/appcache/appcache_group.h24
-rw-r--r--webkit/appcache/appcache_group_unittest.cc63
-rw-r--r--webkit/appcache/appcache_host.h1
-rw-r--r--webkit/appcache/appcache_update_job.cc2
-rw-r--r--webkit/appcache/appcache_update_job.h2
-rw-r--r--webkit/appcache/appcache_update_job_unittest.cc50
7 files changed, 256 insertions, 3 deletions
diff --git a/webkit/appcache/appcache_group.cc b/webkit/appcache/appcache_group.cc
index 49c8af3..a9ad808 100644
--- a/webkit/appcache/appcache_group.cc
+++ b/webkit/appcache/appcache_group.cc
@@ -7,6 +7,7 @@
#include <algorithm>
#include "base/logging.h"
+#include "base/message_loop.h"
#include "webkit/appcache/appcache.h"
#include "webkit/appcache/appcache_host.h"
#include "webkit/appcache/appcache_service.h"
@@ -15,6 +16,24 @@
namespace appcache {
+class AppCacheGroup;
+
+// Use this helper class because we cannot make AppCacheGroup a derived class
+// of AppCacheHost::Observer as it would create a circular dependency between
+// AppCacheHost and AppCacheGroup.
+class AppCacheGroup::HostObserver : public AppCacheHost::Observer {
+ public:
+ explicit HostObserver(AppCacheGroup* group) : group_(group) {}
+
+ // Methods for AppCacheHost::Observer.
+ void OnCacheSelectionComplete(AppCacheHost* host) {} // N/A
+ void OnDestructionImminent(AppCacheHost* host) {
+ group_->HostDestructionImminent(host);
+ }
+ private:
+ AppCacheGroup* group_;
+};
+
AppCacheGroup::AppCacheGroup(AppCacheService* service,
const GURL& manifest_url,
int64 group_id)
@@ -24,13 +43,17 @@ AppCacheGroup::AppCacheGroup(AppCacheService* service,
is_obsolete_(false),
newest_complete_cache_(NULL),
update_job_(NULL),
- service_(service) {
+ service_(service),
+ restart_update_task_(NULL) {
service_->storage()->working_set()->AddGroup(this);
+ host_observer_.reset(new HostObserver(this));
}
AppCacheGroup::~AppCacheGroup() {
DCHECK(old_caches_.empty());
DCHECK(!newest_complete_cache_);
+ DCHECK(!restart_update_task_);
+ DCHECK(queued_updates_.empty());
if (update_job_)
delete update_job_;
@@ -40,11 +63,18 @@ AppCacheGroup::~AppCacheGroup() {
}
void AppCacheGroup::AddUpdateObserver(UpdateObserver* observer) {
- observers_.AddObserver(observer);
+ // If observer being added is a host that has been queued for later update,
+ // add observer to a different observer list.
+ AppCacheHost* host = static_cast<AppCacheHost*>(observer);
+ if (queued_updates_.find(host) != queued_updates_.end())
+ queued_observers_.AddObserver(observer);
+ else
+ observers_.AddObserver(observer);
}
void AppCacheGroup::RemoveUpdateObserver(UpdateObserver* observer) {
observers_.RemoveObserver(observer);
+ queued_observers_.RemoveObserver(observer);
}
void AppCacheGroup::AddCache(AppCache* complete_cache) {
@@ -97,6 +127,79 @@ void AppCacheGroup::StartUpdateWithNewMasterEntry(
update_job_ = new AppCacheUpdateJob(service_, this);
update_job_->StartUpdate(host, new_master_resource);
+
+ // Run queued update immediately as an update has been started manually.
+ if (restart_update_task_) {
+ restart_update_task_->Cancel();
+ restart_update_task_ = NULL;
+ RunQueuedUpdates();
+ }
+}
+
+void AppCacheGroup::QueueUpdate(AppCacheHost* host,
+ const GURL& new_master_resource) {
+ DCHECK(update_job_ && host && !new_master_resource.is_empty());
+ queued_updates_.insert(QueuedUpdates::value_type(host, new_master_resource));
+
+ // Need to know when host is destroyed.
+ host->AddObserver(host_observer_.get());
+
+ // If host is already observing for updates, move host to queued observers
+ // list so that host is not notified when the current update completes.
+ if (FindObserver(host, observers_)) {
+ observers_.RemoveObserver(host);
+ queued_observers_.AddObserver(host);
+ }
+}
+
+void AppCacheGroup::RunQueuedUpdates() {
+ if (restart_update_task_)
+ restart_update_task_ = NULL;
+
+ if (queued_updates_.empty())
+ return;
+
+ QueuedUpdates updates_to_run;
+ queued_updates_.swap(updates_to_run);
+ DCHECK(queued_updates_.empty());
+
+ for (QueuedUpdates::iterator it = updates_to_run.begin();
+ it != updates_to_run.end(); ++it) {
+ AppCacheHost* host = it->first;
+ host->RemoveObserver(host_observer_.get());
+ if (FindObserver(host, queued_observers_)) {
+ queued_observers_.RemoveObserver(host);
+ observers_.AddObserver(host);
+ }
+ StartUpdateWithNewMasterEntry(host, it->second);
+ }
+}
+
+bool AppCacheGroup::FindObserver(UpdateObserver* find_me,
+ const ObserverList<UpdateObserver>& observer_list) {
+ ObserverList<UpdateObserver>::Iterator it(observer_list);
+ UpdateObserver* obs;
+ while ((obs = it.GetNext()) != NULL) {
+ if (obs == find_me)
+ return true;
+ }
+ return false;
+}
+
+void AppCacheGroup::ScheduleUpdateRestart(int delay_ms) {
+ DCHECK(!restart_update_task_);
+ restart_update_task_ =
+ NewRunnableMethod(this, &AppCacheGroup::RunQueuedUpdates);
+ MessageLoop::current()->PostDelayedTask(FROM_HERE, restart_update_task_,
+ delay_ms);
+}
+
+void AppCacheGroup::HostDestructionImminent(AppCacheHost* host) {
+ queued_updates_.erase(host);
+ if (queued_updates_.empty() && restart_update_task_) {
+ restart_update_task_->Cancel();
+ restart_update_task_ = NULL;
+ }
}
void AppCacheGroup::SetUpdateStatus(UpdateStatus status) {
@@ -109,7 +212,17 @@ void AppCacheGroup::SetUpdateStatus(UpdateStatus status) {
DCHECK(update_job_);
} else {
update_job_ = NULL;
+
+ // Check member variable before notifying observers about update finishing.
+ // Observers may remove reference to group, causing group to be deleted
+ // after the notifications. If there are queued updates, then the group
+ // will continue to exist.
+ bool restart_update = !queued_updates_.empty();
+
FOR_EACH_OBSERVER(UpdateObserver, observers_, OnUpdateComplete(this));
+
+ if (restart_update)
+ ScheduleUpdateRestart(kUpdateRestartDelayMs);
}
}
diff --git a/webkit/appcache/appcache_group.h b/webkit/appcache/appcache_group.h
index efe8049..5cf3d99 100644
--- a/webkit/appcache/appcache_group.h
+++ b/webkit/appcache/appcache_group.h
@@ -5,10 +5,13 @@
#ifndef WEBKIT_APPCACHE_APPCACHE_GROUP_H_
#define WEBKIT_APPCACHE_APPCACHE_GROUP_H_
+#include <map>
#include <vector>
#include "base/observer_list.h"
#include "base/ref_counted.h"
+#include "base/scoped_ptr.h"
+#include "base/task.h"
#include "googleurl/src/gurl.h"
#include "testing/gtest/include/gtest/gtest_prod.h"
@@ -18,6 +21,7 @@ class AppCache;
class AppCacheHost;
class AppCacheService;
class AppCacheUpdateJob;
+class HostObserver;
// Collection of application caches identified by the same manifest URL.
// A group exists as long as it is in use by a host or is being updated.
@@ -76,6 +80,8 @@ class AppCacheGroup : public base::RefCounted<AppCacheGroup> {
const GURL& new_master_resource);
private:
+ class HostObserver;
+
friend class AppCacheUpdateJob;
friend class AppCacheUpdateJobTest;
friend class base::RefCounted<AppCacheGroup>;
@@ -84,12 +90,23 @@ class AppCacheGroup : public base::RefCounted<AppCacheGroup> {
~AppCacheGroup();
typedef std::vector<AppCache*> Caches;
+ typedef std::map<AppCacheHost*, GURL> QueuedUpdates;
+
+ static const int kUpdateRestartDelayMs = 1000;
AppCacheUpdateJob* update_job() { return update_job_; }
void SetUpdateStatus(UpdateStatus status);
const Caches& old_caches() const { return old_caches_; }
+ // Update cannot be processed at this time. Queue it for a later run.
+ void QueueUpdate(AppCacheHost* host, const GURL& new_master_resource);
+ void RunQueuedUpdates();
+ bool FindObserver(UpdateObserver* find_me,
+ const ObserverList<UpdateObserver>& observer_list);
+ void ScheduleUpdateRestart(int delay_ms);
+ void HostDestructionImminent(AppCacheHost* host);
+
const int64 group_id_;
const GURL manifest_url_;
UpdateStatus update_status_;
@@ -110,8 +127,15 @@ class AppCacheGroup : public base::RefCounted<AppCacheGroup> {
// List of objects observing this group.
ObserverList<UpdateObserver> observers_;
+ // Updates that have been queued for the next run.
+ QueuedUpdates queued_updates_;
+ ObserverList<UpdateObserver> queued_observers_;
+ CancelableTask* restart_update_task_;
+ scoped_ptr<HostObserver> host_observer_;
+
FRIEND_TEST(AppCacheGroupTest, StartUpdate);
FRIEND_TEST(AppCacheGroupTest, CancelUpdate);
+ FRIEND_TEST(AppCacheGroupTest, QueueUpdate);
FRIEND_TEST(AppCacheUpdateJobTest, AlreadyChecking);
FRIEND_TEST(AppCacheUpdateJobTest, AlreadyDownloading);
DISALLOW_COPY_AND_ASSIGN(AppCacheGroup);
diff --git a/webkit/appcache/appcache_group_unittest.cc b/webkit/appcache/appcache_group_unittest.cc
index 81280e7..384a38b 100644
--- a/webkit/appcache/appcache_group_unittest.cc
+++ b/webkit/appcache/appcache_group_unittest.cc
@@ -56,6 +56,21 @@ class TestUpdateObserver : public AppCacheGroup::UpdateObserver {
bool group_has_cache_;
};
+class TestAppCacheHost : public AppCacheHost {
+ public:
+ TestAppCacheHost(int host_id, AppCacheFrontend* frontend,
+ AppCacheService* service)
+ : AppCacheHost(host_id, frontend, service),
+ update_completed_(false) {
+ }
+
+ virtual void OnUpdateComplete(AppCacheGroup* group) {
+ update_completed_ = true;
+ }
+
+ bool update_completed_;
+};
+
class AppCacheGroupTest : public testing::Test {
};
@@ -208,4 +223,52 @@ TEST(AppCacheGroupTest, CancelUpdate) {
EXPECT_FALSE(observer.group_has_cache_);
}
+TEST(AppCacheGroupTest, QueueUpdate) {
+ MockAppCacheService service;
+ scoped_refptr<AppCacheGroup> group =
+ new AppCacheGroup(&service, GURL("http://foo.com"), 111);
+
+ // Set state to checking to prevent update job from executing fetches.
+ group->update_status_ = AppCacheGroup::CHECKING;
+ group->StartUpdate();
+ EXPECT_TRUE(group->update_job_);
+
+ // Pretend group's update job is terminating so that next update is queued.
+ group->update_job_->internal_state_ = AppCacheUpdateJob::REFETCH_MANIFEST;
+ EXPECT_TRUE(group->update_job_->IsTerminating());
+
+ TestAppCacheFrontend frontend;
+ TestAppCacheHost host(1, &frontend, &service);
+ host.new_master_entry_url_ = GURL("http://foo.com/bar.txt");
+ group->StartUpdateWithNewMasterEntry(&host, host.new_master_entry_url_);
+ EXPECT_FALSE(group->queued_updates_.empty());
+
+ group->AddUpdateObserver(&host);
+ EXPECT_FALSE(group->FindObserver(&host, group->observers_));
+ EXPECT_TRUE(group->FindObserver(&host, group->queued_observers_));
+
+ // Delete update to cause it to complete. Verify no update complete notice
+ // sent to host.
+ delete group->update_job_;
+ EXPECT_EQ(AppCacheGroup::IDLE, group->update_status_);
+ EXPECT_TRUE(group->restart_update_task_);
+ EXPECT_FALSE(host.update_completed_);
+
+ // Start another update. Cancels task and will run queued updates.
+ group->update_status_ = AppCacheGroup::CHECKING; // prevent actual fetches
+ group->StartUpdate();
+ EXPECT_TRUE(group->update_job_);
+ EXPECT_FALSE(group->restart_update_task_);
+ EXPECT_TRUE(group->queued_updates_.empty());
+ EXPECT_FALSE(group->update_job_->pending_master_entries_.empty());
+ EXPECT_FALSE(group->FindObserver(&host, group->queued_observers_));
+ EXPECT_TRUE(group->FindObserver(&host, group->observers_));
+
+ // Delete update to cause it to complete. Verify host is notified.
+ delete group->update_job_;
+ EXPECT_EQ(AppCacheGroup::IDLE, group->update_status_);
+ EXPECT_FALSE(group->restart_update_task_);
+ EXPECT_TRUE(host.update_completed_);
+}
+
} // namespace appcache
diff --git a/webkit/appcache/appcache_host.h b/webkit/appcache/appcache_host.h
index f154bae..13acdb0 100644
--- a/webkit/appcache/appcache_host.h
+++ b/webkit/appcache/appcache_host.h
@@ -178,6 +178,7 @@ class AppCacheHost : public AppCacheStorage::Delegate,
FRIEND_TEST(AppCacheHostTest, FailedCacheLoad);
FRIEND_TEST(AppCacheHostTest, FailedGroupLoad);
FRIEND_TEST(AppCacheHostTest, SetSwappableCache);
+ FRIEND_TEST(AppCacheGroupTest, QueueUpdate);
DISALLOW_COPY_AND_ASSIGN(AppCacheHost);
};
diff --git a/webkit/appcache/appcache_update_job.cc b/webkit/appcache/appcache_update_job.cc
index 8cd752f..8d7ddc1 100644
--- a/webkit/appcache/appcache_update_job.cc
+++ b/webkit/appcache/appcache_update_job.cc
@@ -142,7 +142,7 @@ void AppCacheUpdateJob::StartUpdate(AppCacheHost* host,
// Cannot add more to this update if already terminating.
if (IsTerminating()) {
- // TODO(jennb): requeue in group
+ group_->QueueUpdate(host, new_master_resource);
return;
}
diff --git a/webkit/appcache/appcache_update_job.h b/webkit/appcache/appcache_update_job.h
index 7c60ef1..304df03 100644
--- a/webkit/appcache/appcache_update_job.h
+++ b/webkit/appcache/appcache_update_job.h
@@ -15,6 +15,7 @@
#include "base/task.h"
#include "googleurl/src/gurl.h"
#include "net/url_request/url_request.h"
+#include "testing/gtest/include/gtest/gtest_prod.h"
#include "webkit/appcache/appcache.h"
#include "webkit/appcache/appcache_host.h"
#include "webkit/appcache/appcache_interfaces.h"
@@ -234,6 +235,7 @@ class AppCacheUpdateJob : public URLRequest::Delegate,
net::CompletionCallbackImpl<AppCacheUpdateJob> manifest_data_write_callback_;
net::CompletionCallbackImpl<AppCacheUpdateJob> manifest_data_read_callback_;
+ FRIEND_TEST(AppCacheGroupTest, QueueUpdate);
DISALLOW_COPY_AND_ASSIGN(AppCacheUpdateJob);
};
diff --git a/webkit/appcache/appcache_update_job_unittest.cc b/webkit/appcache/appcache_update_job_unittest.cc
index 70b980a..5679ef3 100644
--- a/webkit/appcache/appcache_update_job_unittest.cc
+++ b/webkit/appcache/appcache_update_job_unittest.cc
@@ -1890,6 +1890,52 @@ class AppCacheUpdateJobTest : public testing::Test,
WaitForUpdateToFinish();
}
+ void QueueMasterEntryTest() {
+ ASSERT_EQ(MessageLoop::TYPE_IO, MessageLoop::current()->type());
+
+ MakeService();
+ group_ = new AppCacheGroup(
+ service_.get(), http_server_->TestServerPage("files/manifest1"), 111);
+ AppCacheUpdateJob* update = new AppCacheUpdateJob(service_.get(), group_);
+ group_->update_job_ = update;
+
+ // Pretend update job has been running and is about to terminate.
+ group_->update_status_ = AppCacheGroup::DOWNLOADING;
+ update->internal_state_ = AppCacheUpdateJob::REFETCH_MANIFEST;
+ EXPECT_TRUE(update->IsTerminating());
+
+ // Start an update. Should be queued.
+ MockFrontend* frontend = MakeMockFrontend();
+ AppCacheHost* host = MakeHost(1, frontend);
+ host->new_master_entry_url_ =
+ http_server_->TestServerPage("files/explicit2");
+ update->StartUpdate(host, host->new_master_entry_url_);
+ EXPECT_TRUE(update->pending_master_entries_.empty());
+ EXPECT_FALSE(group_->queued_updates_.empty());
+
+ // Delete update, causing it to finish, which should trigger a new update
+ // for the queued host and master entry after a delay.
+ delete update;
+ EXPECT_TRUE(group_->restart_update_task_);
+
+ // Set up checks for when queued update job finishes.
+ do_checks_after_update_finished_ = true;
+ expect_group_obsolete_ = false;
+ expect_group_has_cache_ = true;
+ tested_manifest_ = MANIFEST1;
+ expect_extra_entries_.insert(AppCache::EntryMap::value_type(
+ host->new_master_entry_url_, AppCacheEntry(AppCacheEntry::MASTER)));
+ MockFrontend::HostIds ids1(1, host->host_id());
+ frontend->AddExpectedEvent(ids1, CHECKING_EVENT);
+ frontend->AddExpectedEvent(ids1, DOWNLOADING_EVENT);
+ frontend->AddExpectedEvent(ids1, PROGRESS_EVENT);
+ frontend->AddExpectedEvent(ids1, PROGRESS_EVENT);
+ frontend->AddExpectedEvent(ids1, CACHED_EVENT);
+
+ // Group status will be IDLE so cannot call WaitForUpdateToFinish.
+ group_->AddUpdateObserver(this);
+ }
+
void WaitForUpdateToFinish() {
if (group_->update_status() == AppCacheGroup::IDLE)
UpdateFinished();
@@ -2458,4 +2504,8 @@ TEST_F(AppCacheUpdateJobTest, StartUpdateMidDownload) {
RunTestOnIOThread(&AppCacheUpdateJobTest::StartUpdateMidDownloadTest);
}
+TEST_F(AppCacheUpdateJobTest, QueueMasterEntry) {
+ RunTestOnIOThread(&AppCacheUpdateJobTest::QueueMasterEntryTest);
+}
+
} // namespace appcache