blob: 00e3aa75bed71ad59a650fd69c8d64223b77171b [file] [log] [blame]
// 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 "content/browser/service_worker/service_worker_storage.h"
#include <stddef.h>
#include <memory>
#include <utility>
#include "base/bind_helpers.h"
#include "base/files/file_util.h"
#include "base/memory/ptr_util.h"
#include "base/message_loop/message_loop.h"
#include "base/sequenced_task_runner.h"
#include "base/task_runner_util.h"
#include "base/task_scheduler/post_task.h"
#include "base/trace_event/trace_event.h"
#include "content/browser/service_worker/service_worker_context_core.h"
#include "content/browser/service_worker/service_worker_disk_cache.h"
#include "content/browser/service_worker/service_worker_info.h"
#include "content/browser/service_worker/service_worker_registration.h"
#include "content/browser/service_worker/service_worker_version.h"
#include "content/common/service_worker/service_worker_types.h"
#include "content/common/service_worker/service_worker_utils.h"
#include "content/public/browser/browser_thread.h"
#include "net/base/completion_callback.h"
#include "net/base/io_buffer.h"
#include "net/base/net_errors.h"
#include "storage/browser/quota/quota_manager_proxy.h"
#include "storage/browser/quota/special_storage_policy.h"
#include "third_party/WebKit/common/quota/quota_types.mojom.h"
#include "third_party/WebKit/common/service_worker/service_worker_object.mojom.h"
#include "third_party/WebKit/common/service_worker/service_worker_registration.mojom.h"
#include "third_party/WebKit/public/platform/web_feature.mojom.h"
using std::swap;
namespace content {
namespace {
void RunSoon(const base::Location& from_here, base::OnceClosure closure) {
base::ThreadTaskRunnerHandle::Get()->PostTask(from_here, std::move(closure));
}
void CompleteFindNow(scoped_refptr<ServiceWorkerRegistration> registration,
ServiceWorkerStatusCode status,
ServiceWorkerStorage::FindRegistrationCallback callback) {
if (registration && registration->is_deleted()) {
// It's past the point of no return and no longer findable.
std::move(callback).Run(SERVICE_WORKER_ERROR_NOT_FOUND, nullptr);
return;
}
std::move(callback).Run(status, std::move(registration));
}
void CompleteFindSoon(const base::Location& from_here,
scoped_refptr<ServiceWorkerRegistration> registration,
ServiceWorkerStatusCode status,
ServiceWorkerStorage::FindRegistrationCallback callback) {
RunSoon(from_here, base::BindOnce(&CompleteFindNow, std::move(registration),
status, std::move(callback)));
}
const base::FilePath::CharType kDatabaseName[] =
FILE_PATH_LITERAL("Database");
const base::FilePath::CharType kDiskCacheName[] =
FILE_PATH_LITERAL("ScriptCache");
const int kMaxServiceWorkerStorageMemDiskCacheSize = 10 * 1024 * 1024;
const int kMaxServiceWorkerStorageDiskCacheSize = 250 * 1024 * 1024;
ServiceWorkerStatusCode DatabaseStatusToStatusCode(
ServiceWorkerDatabase::Status status) {
switch (status) {
case ServiceWorkerDatabase::STATUS_OK:
return SERVICE_WORKER_OK;
case ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND:
return SERVICE_WORKER_ERROR_NOT_FOUND;
case ServiceWorkerDatabase::STATUS_ERROR_MAX:
NOTREACHED();
default:
return SERVICE_WORKER_ERROR_FAILED;
}
}
void DidUpdateNavigationPreloadState(
ServiceWorkerStorage::StatusCallback callback,
ServiceWorkerDatabase::Status status) {
std::move(callback).Run(DatabaseStatusToStatusCode(status));
}
} // namespace
ServiceWorkerStorage::InitialData::InitialData()
: next_registration_id(blink::mojom::kInvalidServiceWorkerRegistrationId),
next_version_id(blink::mojom::kInvalidServiceWorkerVersionId),
next_resource_id(kInvalidServiceWorkerResourceId) {}
ServiceWorkerStorage::InitialData::~InitialData() {
}
ServiceWorkerStorage::DidDeleteRegistrationParams::DidDeleteRegistrationParams(
int64_t registration_id,
GURL origin,
StatusCallback callback)
: registration_id(registration_id),
origin(origin),
callback(std::move(callback)) {}
ServiceWorkerStorage::DidDeleteRegistrationParams::
~DidDeleteRegistrationParams() {}
ServiceWorkerStorage::~ServiceWorkerStorage() {
ClearSessionOnlyOrigins();
weak_factory_.InvalidateWeakPtrs();
database_task_runner_->DeleteSoon(FROM_HERE, std::move(database_));
}
// static
std::unique_ptr<ServiceWorkerStorage> ServiceWorkerStorage::Create(
const base::FilePath& path,
const base::WeakPtr<ServiceWorkerContextCore>& context,
scoped_refptr<base::SequencedTaskRunner> database_task_runner,
storage::QuotaManagerProxy* quota_manager_proxy,
storage::SpecialStoragePolicy* special_storage_policy) {
return base::WrapUnique(
new ServiceWorkerStorage(path, context, std::move(database_task_runner),
quota_manager_proxy, special_storage_policy));
}
// static
std::unique_ptr<ServiceWorkerStorage> ServiceWorkerStorage::Create(
const base::WeakPtr<ServiceWorkerContextCore>& context,
ServiceWorkerStorage* old_storage) {
return base::WrapUnique(new ServiceWorkerStorage(
old_storage->path_, context, old_storage->database_task_runner_,
old_storage->quota_manager_proxy_.get(),
old_storage->special_storage_policy_.get()));
}
void ServiceWorkerStorage::FindRegistrationForDocument(
const GURL& document_url,
FindRegistrationCallback callback) {
DCHECK(!document_url.has_ref());
switch (state_) {
case DISABLED:
CompleteFindNow(scoped_refptr<ServiceWorkerRegistration>(),
SERVICE_WORKER_ERROR_ABORT, std::move(callback));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(base::BindOnce(
&ServiceWorkerStorage::FindRegistrationForDocument,
weak_factory_.GetWeakPtr(), document_url, std::move(callback)));
TRACE_EVENT_INSTANT1(
"ServiceWorker",
"ServiceWorkerStorage::FindRegistrationForDocument:LazyInitialize",
TRACE_EVENT_SCOPE_THREAD, "URL", document_url.spec());
return;
case INITIALIZED:
break;
}
// See if there are any stored registrations for the origin.
if (!base::ContainsKey(registered_origins_, document_url.GetOrigin())) {
// Look for something currently being installed.
scoped_refptr<ServiceWorkerRegistration> installing_registration =
FindInstallingRegistrationForDocument(document_url);
ServiceWorkerStatusCode status = installing_registration
? SERVICE_WORKER_OK
: SERVICE_WORKER_ERROR_NOT_FOUND;
TRACE_EVENT_INSTANT2(
"ServiceWorker",
"ServiceWorkerStorage::FindRegistrationForDocument:CheckInstalling",
TRACE_EVENT_SCOPE_THREAD,
"URL", document_url.spec(),
"Status", ServiceWorkerStatusToString(status));
CompleteFindNow(std::move(installing_registration), status,
std::move(callback));
return;
}
// To connect this TRACE_EVENT with the callback, TimeTicks is used for
// callback id.
int64_t callback_id = base::TimeTicks::Now().ToInternalValue();
TRACE_EVENT_ASYNC_BEGIN1("ServiceWorker",
"ServiceWorkerStorage::FindRegistrationForDocument",
callback_id, "URL", document_url.spec());
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&FindForDocumentInDB, database_.get(),
base::ThreadTaskRunnerHandle::Get(), document_url,
base::BindOnce(&ServiceWorkerStorage::DidFindRegistrationForDocument,
weak_factory_.GetWeakPtr(), document_url,
std::move(callback), callback_id)));
}
void ServiceWorkerStorage::FindRegistrationForPattern(
const GURL& scope,
FindRegistrationCallback callback) {
switch (state_) {
case DISABLED:
CompleteFindSoon(FROM_HERE, scoped_refptr<ServiceWorkerRegistration>(),
SERVICE_WORKER_ERROR_ABORT, std::move(callback));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(base::BindOnce(
&ServiceWorkerStorage::FindRegistrationForPattern,
weak_factory_.GetWeakPtr(), scope, std::move(callback)));
return;
case INITIALIZED:
break;
}
// See if there are any stored registrations for the origin.
if (!base::ContainsKey(registered_origins_, scope.GetOrigin())) {
// Look for something currently being installed.
scoped_refptr<ServiceWorkerRegistration> installing_registration =
FindInstallingRegistrationForPattern(scope);
ServiceWorkerStatusCode installing_status =
installing_registration ? SERVICE_WORKER_OK
: SERVICE_WORKER_ERROR_NOT_FOUND;
CompleteFindSoon(FROM_HERE, std::move(installing_registration),
installing_status, std::move(callback));
return;
}
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&FindForPatternInDB, database_.get(),
base::ThreadTaskRunnerHandle::Get(), scope,
base::BindOnce(&ServiceWorkerStorage::DidFindRegistrationForPattern,
weak_factory_.GetWeakPtr(), scope,
std::move(callback))));
}
ServiceWorkerRegistration* ServiceWorkerStorage::GetUninstallingRegistration(
const GURL& scope) {
if (state_ != INITIALIZED)
return nullptr;
for (const auto& registration : uninstalling_registrations_) {
if (registration.second->pattern() == scope) {
DCHECK(registration.second->is_uninstalling());
return registration.second.get();
}
}
return nullptr;
}
void ServiceWorkerStorage::FindRegistrationForId(
int64_t registration_id,
const GURL& origin,
FindRegistrationCallback callback) {
switch (state_) {
case DISABLED:
CompleteFindNow(scoped_refptr<ServiceWorkerRegistration>(),
SERVICE_WORKER_ERROR_ABORT, std::move(callback));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(
base::BindOnce(&ServiceWorkerStorage::FindRegistrationForId,
weak_factory_.GetWeakPtr(), registration_id, origin,
std::move(callback)));
return;
case INITIALIZED:
break;
}
// See if there are any stored registrations for the origin.
if (!base::ContainsKey(registered_origins_, origin)) {
// Look for something currently being installed.
scoped_refptr<ServiceWorkerRegistration> installing_registration =
FindInstallingRegistrationForId(registration_id);
CompleteFindNow(installing_registration,
installing_registration ? SERVICE_WORKER_OK
: SERVICE_WORKER_ERROR_NOT_FOUND,
std::move(callback));
return;
}
scoped_refptr<ServiceWorkerRegistration> registration =
context_->GetLiveRegistration(registration_id);
if (registration) {
CompleteFindNow(std::move(registration), SERVICE_WORKER_OK,
std::move(callback));
return;
}
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&FindForIdInDB, database_.get(), base::ThreadTaskRunnerHandle::Get(),
registration_id, origin,
base::BindOnce(&ServiceWorkerStorage::DidFindRegistrationForId,
weak_factory_.GetWeakPtr(), std::move(callback))));
}
void ServiceWorkerStorage::FindRegistrationForIdOnly(
int64_t registration_id,
FindRegistrationCallback callback) {
switch (state_) {
case DISABLED:
CompleteFindNow(nullptr, SERVICE_WORKER_ERROR_ABORT, std::move(callback));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(base::BindOnce(
&ServiceWorkerStorage::FindRegistrationForIdOnly,
weak_factory_.GetWeakPtr(), registration_id, std::move(callback)));
return;
case INITIALIZED:
break;
}
scoped_refptr<ServiceWorkerRegistration> registration =
context_->GetLiveRegistration(registration_id);
if (registration) {
// Delegate to FindRegistrationForId to make sure the same subset of live
// registrations is returned.
// TODO(mek): CompleteFindNow should really do all the required checks, so
// calling that directly here should be enough.
FindRegistrationForId(registration_id, registration->pattern().GetOrigin(),
std::move(callback));
return;
}
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&FindForIdOnlyInDB, database_.get(),
base::ThreadTaskRunnerHandle::Get(), registration_id,
base::BindOnce(&ServiceWorkerStorage::DidFindRegistrationForId,
weak_factory_.GetWeakPtr(), std::move(callback))));
}
void ServiceWorkerStorage::GetRegistrationsForOrigin(
const GURL& origin,
GetRegistrationsCallback callback) {
switch (state_) {
case DISABLED:
RunSoon(FROM_HERE,
base::BindOnce(
std::move(callback), SERVICE_WORKER_ERROR_ABORT,
std::vector<scoped_refptr<ServiceWorkerRegistration>>()));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(base::BindOnce(
&ServiceWorkerStorage::GetRegistrationsForOrigin,
weak_factory_.GetWeakPtr(), origin, std::move(callback)));
return;
case INITIALIZED:
break;
}
RegistrationList* registrations = new RegistrationList;
std::vector<ResourceList>* resource_lists = new std::vector<ResourceList>;
base::PostTaskAndReplyWithResult(
database_task_runner_.get(), FROM_HERE,
base::BindOnce(&ServiceWorkerDatabase::GetRegistrationsForOrigin,
base::Unretained(database_.get()), origin, registrations,
resource_lists),
base::BindOnce(&ServiceWorkerStorage::DidGetRegistrationsForOrigin,
weak_factory_.GetWeakPtr(), std::move(callback),
base::Owned(registrations), base::Owned(resource_lists),
origin));
}
void ServiceWorkerStorage::GetAllRegistrationsInfos(
GetRegistrationsInfosCallback callback) {
switch (state_) {
case DISABLED:
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_ABORT,
std::vector<ServiceWorkerRegistrationInfo>()));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(
base::BindOnce(&ServiceWorkerStorage::GetAllRegistrationsInfos,
weak_factory_.GetWeakPtr(), std::move(callback)));
return;
case INITIALIZED:
break;
}
RegistrationList* registrations = new RegistrationList;
base::PostTaskAndReplyWithResult(
database_task_runner_.get(), FROM_HERE,
base::BindOnce(&ServiceWorkerDatabase::GetAllRegistrations,
base::Unretained(database_.get()), registrations),
base::BindOnce(&ServiceWorkerStorage::DidGetAllRegistrationsInfos,
weak_factory_.GetWeakPtr(), std::move(callback),
base::Owned(registrations)));
}
void ServiceWorkerStorage::StoreRegistration(
ServiceWorkerRegistration* registration,
ServiceWorkerVersion* version,
StatusCallback callback) {
DCHECK(registration);
DCHECK(version);
DCHECK(state_ == INITIALIZED || state_ == DISABLED) << state_;
if (IsDisabled()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_ABORT));
return;
}
DCHECK_NE(version->fetch_handler_existence(),
ServiceWorkerVersion::FetchHandlerExistence::UNKNOWN);
ServiceWorkerDatabase::RegistrationData data;
data.registration_id = registration->id();
data.scope = registration->pattern();
data.update_via_cache = registration->update_via_cache();
data.script = version->script_url();
data.has_fetch_handler = version->fetch_handler_existence() ==
ServiceWorkerVersion::FetchHandlerExistence::EXISTS;
data.version_id = version->version_id();
data.last_update_check = registration->last_update_check();
data.is_active = (version == registration->active_version());
if (version->origin_trial_tokens())
data.origin_trial_tokens = *version->origin_trial_tokens();
data.navigation_preload_state = registration->navigation_preload_state();
data.used_features = version->used_features();
ResourceList resources;
version->script_cache_map()->GetResources(&resources);
if (resources.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_FAILED));
return;
}
uint64_t resources_total_size_bytes = 0;
for (const auto& resource : resources) {
resources_total_size_bytes += resource.size_bytes;
}
data.resources_total_size_bytes = resources_total_size_bytes;
if (!has_checked_for_stale_resources_)
DeleteStaleResources();
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&WriteRegistrationInDB, database_.get(),
base::ThreadTaskRunnerHandle::Get(), data, resources,
base::BindOnce(&ServiceWorkerStorage::DidStoreRegistration,
weak_factory_.GetWeakPtr(),
std::move(callback), data)));
registration->set_is_deleted(false);
}
void ServiceWorkerStorage::UpdateToActiveState(
ServiceWorkerRegistration* registration,
StatusCallback callback) {
DCHECK(registration);
DCHECK(state_ == INITIALIZED || state_ == DISABLED) << state_;
if (IsDisabled()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_ABORT));
return;
}
base::PostTaskAndReplyWithResult(
database_task_runner_.get(), FROM_HERE,
base::BindOnce(&ServiceWorkerDatabase::UpdateVersionToActive,
base::Unretained(database_.get()), registration->id(),
registration->pattern().GetOrigin()),
base::BindOnce(&ServiceWorkerStorage::DidUpdateToActiveState,
weak_factory_.GetWeakPtr(), std::move(callback)));
}
void ServiceWorkerStorage::UpdateLastUpdateCheckTime(
ServiceWorkerRegistration* registration) {
DCHECK(registration);
DCHECK(state_ == INITIALIZED || state_ == DISABLED) << state_;
if (IsDisabled())
return;
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
base::IgnoreResult(&ServiceWorkerDatabase::UpdateLastCheckTime),
base::Unretained(database_.get()), registration->id(),
registration->pattern().GetOrigin(),
registration->last_update_check()));
}
void ServiceWorkerStorage::UpdateNavigationPreloadEnabled(
int64_t registration_id,
const GURL& origin,
bool enable,
StatusCallback callback) {
DCHECK(state_ == INITIALIZED || state_ == DISABLED) << state_;
if (IsDisabled()) {
std::move(callback).Run(SERVICE_WORKER_ERROR_ABORT);
return;
}
base::PostTaskAndReplyWithResult(
database_task_runner_.get(), FROM_HERE,
base::BindOnce(&ServiceWorkerDatabase::UpdateNavigationPreloadEnabled,
base::Unretained(database_.get()), registration_id, origin,
enable),
base::BindOnce(&DidUpdateNavigationPreloadState, std::move(callback)));
}
void ServiceWorkerStorage::UpdateNavigationPreloadHeader(
int64_t registration_id,
const GURL& origin,
const std::string& value,
StatusCallback callback) {
DCHECK(state_ == INITIALIZED || state_ == DISABLED) << state_;
if (IsDisabled()) {
std::move(callback).Run(SERVICE_WORKER_ERROR_ABORT);
return;
}
base::PostTaskAndReplyWithResult(
database_task_runner_.get(), FROM_HERE,
base::BindOnce(&ServiceWorkerDatabase::UpdateNavigationPreloadHeader,
base::Unretained(database_.get()), registration_id, origin,
value),
base::BindOnce(&DidUpdateNavigationPreloadState, std::move(callback)));
}
void ServiceWorkerStorage::DeleteRegistration(int64_t registration_id,
const GURL& origin,
StatusCallback callback) {
DCHECK(state_ == INITIALIZED || state_ == DISABLED) << state_;
if (IsDisabled()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_ABORT));
return;
}
if (!has_checked_for_stale_resources_)
DeleteStaleResources();
auto params = std::make_unique<DidDeleteRegistrationParams>(
registration_id, origin, std::move(callback));
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&DeleteRegistrationFromDB, database_.get(),
base::ThreadTaskRunnerHandle::Get(), registration_id, origin,
base::BindOnce(&ServiceWorkerStorage::DidDeleteRegistration,
weak_factory_.GetWeakPtr(), std::move(params))));
// The registration should no longer be findable.
pending_deletions_.insert(registration_id);
ServiceWorkerRegistration* registration =
context_->GetLiveRegistration(registration_id);
if (registration)
registration->set_is_deleted(true);
}
std::unique_ptr<ServiceWorkerResponseReader>
ServiceWorkerStorage::CreateResponseReader(int64_t resource_id) {
return base::WrapUnique(
new ServiceWorkerResponseReader(resource_id, disk_cache()->GetWeakPtr()));
}
std::unique_ptr<ServiceWorkerResponseWriter>
ServiceWorkerStorage::CreateResponseWriter(int64_t resource_id) {
return base::WrapUnique(
new ServiceWorkerResponseWriter(resource_id, disk_cache()->GetWeakPtr()));
}
std::unique_ptr<ServiceWorkerResponseMetadataWriter>
ServiceWorkerStorage::CreateResponseMetadataWriter(int64_t resource_id) {
return base::WrapUnique(new ServiceWorkerResponseMetadataWriter(
resource_id, disk_cache()->GetWeakPtr()));
}
void ServiceWorkerStorage::StoreUncommittedResourceId(int64_t resource_id) {
DCHECK_NE(kInvalidServiceWorkerResourceId, resource_id);
DCHECK(INITIALIZED == state_ || DISABLED == state_) << state_;
if (IsDisabled())
return;
if (!has_checked_for_stale_resources_)
DeleteStaleResources();
base::PostTaskAndReplyWithResult(
database_task_runner_.get(), FROM_HERE,
base::BindOnce(&ServiceWorkerDatabase::WriteUncommittedResourceIds,
base::Unretained(database_.get()),
std::set<int64_t>(&resource_id, &resource_id + 1)),
base::BindOnce(&ServiceWorkerStorage::DidWriteUncommittedResourceIds,
weak_factory_.GetWeakPtr()));
}
void ServiceWorkerStorage::DoomUncommittedResource(int64_t resource_id) {
DCHECK_NE(kInvalidServiceWorkerResourceId, resource_id);
DCHECK(INITIALIZED == state_ || DISABLED == state_) << state_;
if (IsDisabled())
return;
DoomUncommittedResources(std::set<int64_t>(&resource_id, &resource_id + 1));
}
void ServiceWorkerStorage::DoomUncommittedResources(
const std::set<int64_t>& resource_ids) {
DCHECK(INITIALIZED == state_ || DISABLED == state_) << state_;
if (IsDisabled())
return;
base::PostTaskAndReplyWithResult(
database_task_runner_.get(), FROM_HERE,
base::BindOnce(&ServiceWorkerDatabase::PurgeUncommittedResourceIds,
base::Unretained(database_.get()), resource_ids),
base::BindOnce(&ServiceWorkerStorage::DidPurgeUncommittedResourceIds,
weak_factory_.GetWeakPtr(), resource_ids));
}
void ServiceWorkerStorage::StoreUserData(
int64_t registration_id,
const GURL& origin,
const std::vector<std::pair<std::string, std::string>>& key_value_pairs,
StatusCallback callback) {
switch (state_) {
case DISABLED:
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_ABORT));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(base::BindOnce(
&ServiceWorkerStorage::StoreUserData, weak_factory_.GetWeakPtr(),
registration_id, origin, key_value_pairs, std::move(callback)));
return;
case INITIALIZED:
break;
}
if (registration_id == blink::mojom::kInvalidServiceWorkerRegistrationId ||
key_value_pairs.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_FAILED));
return;
}
for (const auto& kv : key_value_pairs) {
if (kv.first.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_FAILED));
return;
}
}
base::PostTaskAndReplyWithResult(
database_task_runner_.get(), FROM_HERE,
base::BindOnce(&ServiceWorkerDatabase::WriteUserData,
base::Unretained(database_.get()), registration_id, origin,
key_value_pairs),
base::BindOnce(&ServiceWorkerStorage::DidStoreUserData,
weak_factory_.GetWeakPtr(), std::move(callback)));
}
void ServiceWorkerStorage::GetUserData(int64_t registration_id,
const std::vector<std::string>& keys,
GetUserDataCallback callback) {
switch (state_) {
case DISABLED:
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), std::vector<std::string>(),
SERVICE_WORKER_ERROR_ABORT));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(base::BindOnce(&ServiceWorkerStorage::GetUserData,
weak_factory_.GetWeakPtr(), registration_id,
keys, std::move(callback)));
return;
case INITIALIZED:
break;
}
if (registration_id == blink::mojom::kInvalidServiceWorkerRegistrationId ||
keys.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), std::vector<std::string>(),
SERVICE_WORKER_ERROR_FAILED));
return;
}
for (const std::string& key : keys) {
if (key.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), std::vector<std::string>(),
SERVICE_WORKER_ERROR_FAILED));
return;
}
}
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&ServiceWorkerStorage::GetUserDataInDB, database_.get(),
base::ThreadTaskRunnerHandle::Get(), registration_id, keys,
base::BindOnce(&ServiceWorkerStorage::DidGetUserData,
weak_factory_.GetWeakPtr(), std::move(callback))));
}
void ServiceWorkerStorage::GetUserDataByKeyPrefix(
int64_t registration_id,
const std::string& key_prefix,
GetUserDataCallback callback) {
switch (state_) {
case DISABLED:
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), std::vector<std::string>(),
SERVICE_WORKER_ERROR_ABORT));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(
base::BindOnce(&ServiceWorkerStorage::GetUserDataByKeyPrefix,
weak_factory_.GetWeakPtr(), registration_id,
key_prefix, std::move(callback)));
return;
case INITIALIZED:
break;
}
if (registration_id == blink::mojom::kInvalidServiceWorkerRegistrationId) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), std::vector<std::string>(),
SERVICE_WORKER_ERROR_FAILED));
return;
}
if (key_prefix.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), std::vector<std::string>(),
SERVICE_WORKER_ERROR_FAILED));
return;
}
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&ServiceWorkerStorage::GetUserDataByKeyPrefixInDB, database_.get(),
base::ThreadTaskRunnerHandle::Get(), registration_id, key_prefix,
base::BindOnce(&ServiceWorkerStorage::DidGetUserData,
weak_factory_.GetWeakPtr(), std::move(callback))));
}
void ServiceWorkerStorage::GetUserKeysAndDataByKeyPrefix(
int64_t registration_id,
const std::string& key_prefix,
GetUserKeysAndDataCallback callback) {
switch (state_) {
case DISABLED:
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback),
base::flat_map<std::string, std::string>(),
SERVICE_WORKER_ERROR_ABORT));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(
base::BindOnce(&ServiceWorkerStorage::GetUserKeysAndDataByKeyPrefix,
weak_factory_.GetWeakPtr(), registration_id,
key_prefix, std::move(callback)));
return;
case INITIALIZED:
break;
}
if (registration_id == blink::mojom::kInvalidServiceWorkerRegistrationId ||
key_prefix.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback),
base::flat_map<std::string, std::string>(),
SERVICE_WORKER_ERROR_FAILED));
return;
}
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&ServiceWorkerStorage::GetUserKeysAndDataByKeyPrefixInDB,
database_.get(), base::ThreadTaskRunnerHandle::Get(), registration_id,
key_prefix,
base::BindOnce(&ServiceWorkerStorage::DidGetUserKeysAndData,
weak_factory_.GetWeakPtr(), std::move(callback))));
}
void ServiceWorkerStorage::ClearUserData(int64_t registration_id,
const std::vector<std::string>& keys,
StatusCallback callback) {
switch (state_) {
case DISABLED:
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_ABORT));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(base::BindOnce(&ServiceWorkerStorage::ClearUserData,
weak_factory_.GetWeakPtr(), registration_id,
keys, std::move(callback)));
return;
case INITIALIZED:
break;
}
if (registration_id == blink::mojom::kInvalidServiceWorkerRegistrationId ||
keys.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_FAILED));
return;
}
for (const std::string& key : keys) {
if (key.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_FAILED));
return;
}
}
base::PostTaskAndReplyWithResult(
database_task_runner_.get(), FROM_HERE,
base::BindOnce(&ServiceWorkerDatabase::DeleteUserData,
base::Unretained(database_.get()), registration_id, keys),
base::BindOnce(&ServiceWorkerStorage::DidDeleteUserData,
weak_factory_.GetWeakPtr(), std::move(callback)));
}
void ServiceWorkerStorage::ClearUserDataByKeyPrefixes(
int64_t registration_id,
const std::vector<std::string>& key_prefixes,
StatusCallback callback) {
switch (state_) {
case DISABLED:
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_ABORT));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(
base::BindOnce(&ServiceWorkerStorage::ClearUserDataByKeyPrefixes,
weak_factory_.GetWeakPtr(), registration_id,
key_prefixes, std::move(callback)));
return;
case INITIALIZED:
break;
}
if (registration_id == blink::mojom::kInvalidServiceWorkerRegistrationId ||
key_prefixes.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_FAILED));
return;
}
for (const std::string& key_prefix : key_prefixes) {
if (key_prefix.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback), SERVICE_WORKER_ERROR_FAILED));
return;
}
}
base::PostTaskAndReplyWithResult(
database_task_runner_.get(), FROM_HERE,
base::BindOnce(&ServiceWorkerDatabase::DeleteUserDataByKeyPrefixes,
base::Unretained(database_.get()), registration_id,
key_prefixes),
base::BindOnce(&ServiceWorkerStorage::DidDeleteUserData,
weak_factory_.GetWeakPtr(), std::move(callback)));
}
void ServiceWorkerStorage::GetUserDataForAllRegistrations(
const std::string& key,
GetUserDataForAllRegistrationsCallback callback) {
switch (state_) {
case DISABLED:
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback),
std::vector<std::pair<int64_t, std::string>>(),
SERVICE_WORKER_ERROR_ABORT));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(
base::BindOnce(&ServiceWorkerStorage::GetUserDataForAllRegistrations,
weak_factory_.GetWeakPtr(), key, std::move(callback)));
return;
case INITIALIZED:
break;
}
if (key.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback),
std::vector<std::pair<int64_t, std::string>>(),
SERVICE_WORKER_ERROR_FAILED));
return;
}
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&ServiceWorkerStorage::GetUserDataForAllRegistrationsInDB,
database_.get(), base::ThreadTaskRunnerHandle::Get(), key,
base::BindOnce(
&ServiceWorkerStorage::DidGetUserDataForAllRegistrations,
weak_factory_.GetWeakPtr(), std::move(callback))));
}
void ServiceWorkerStorage::GetUserDataForAllRegistrationsByKeyPrefix(
const std::string& key_prefix,
GetUserDataForAllRegistrationsCallback callback) {
switch (state_) {
case DISABLED:
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback),
std::vector<std::pair<int64_t, std::string>>(),
SERVICE_WORKER_ERROR_ABORT));
return;
case INITIALIZING: // Fall-through.
case UNINITIALIZED:
LazyInitialize(base::BindOnce(
&ServiceWorkerStorage::GetUserDataForAllRegistrationsByKeyPrefix,
weak_factory_.GetWeakPtr(), key_prefix, std::move(callback)));
return;
case INITIALIZED:
break;
}
if (key_prefix.empty()) {
RunSoon(FROM_HERE,
base::BindOnce(std::move(callback),
std::vector<std::pair<int64_t, std::string>>(),
SERVICE_WORKER_ERROR_FAILED));
return;
}
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&ServiceWorkerStorage::GetUserDataForAllRegistrationsByKeyPrefixInDB,
database_.get(), base::ThreadTaskRunnerHandle::Get(), key_prefix,
base::BindOnce(
&ServiceWorkerStorage::DidGetUserDataForAllRegistrations,
weak_factory_.GetWeakPtr(), std::move(callback))));
}
void ServiceWorkerStorage::DeleteAndStartOver(StatusCallback callback) {
Disable();
// Will be used in DiskCacheImplDoneWithDisk()
delete_and_start_over_callback_ = std::move(callback);
// Won't get a callback about cleanup being done, so call it ourselves.
if (!expecting_done_with_disk_on_disable_)
DiskCacheImplDoneWithDisk();
}
void ServiceWorkerStorage::DiskCacheImplDoneWithDisk() {
expecting_done_with_disk_on_disable_ = false;
if (!delete_and_start_over_callback_.is_null()) {
// Delete the database on the database thread.
base::PostTaskAndReplyWithResult(
database_task_runner_.get(), FROM_HERE,
base::BindOnce(&ServiceWorkerDatabase::DestroyDatabase,
base::Unretained(database_.get())),
base::BindOnce(&ServiceWorkerStorage::DidDeleteDatabase,
weak_factory_.GetWeakPtr(),
std::move(delete_and_start_over_callback_)));
}
}
int64_t ServiceWorkerStorage::NewRegistrationId() {
if (state_ == DISABLED)
return blink::mojom::kInvalidServiceWorkerRegistrationId;
DCHECK_EQ(INITIALIZED, state_);
return next_registration_id_++;
}
int64_t ServiceWorkerStorage::NewVersionId() {
if (state_ == DISABLED)
return blink::mojom::kInvalidServiceWorkerVersionId;
DCHECK_EQ(INITIALIZED, state_);
return next_version_id_++;
}
int64_t ServiceWorkerStorage::NewResourceId() {
if (state_ == DISABLED)
return kInvalidServiceWorkerResourceId;
DCHECK_EQ(INITIALIZED, state_);
return next_resource_id_++;
}
void ServiceWorkerStorage::NotifyInstallingRegistration(
ServiceWorkerRegistration* registration) {
DCHECK(installing_registrations_.find(registration->id()) ==
installing_registrations_.end());
installing_registrations_[registration->id()] = registration;
}
void ServiceWorkerStorage::NotifyDoneInstallingRegistration(
ServiceWorkerRegistration* registration,
ServiceWorkerVersion* version,
ServiceWorkerStatusCode status) {
installing_registrations_.erase(registration->id());
if (status != SERVICE_WORKER_OK && version) {
ResourceList resources;
version->script_cache_map()->GetResources(&resources);
std::set<int64_t> resource_ids;
for (const auto& resource : resources)
resource_ids.insert(resource.resource_id);
DoomUncommittedResources(resource_ids);
}
}
void ServiceWorkerStorage::NotifyUninstallingRegistration(
ServiceWorkerRegistration* registration) {
DCHECK(uninstalling_registrations_.find(registration->id()) ==
uninstalling_registrations_.end());
uninstalling_registrations_[registration->id()] = registration;
}
void ServiceWorkerStorage::NotifyDoneUninstallingRegistration(
ServiceWorkerRegistration* registration) {
uninstalling_registrations_.erase(registration->id());
}
void ServiceWorkerStorage::Disable() {
state_ = DISABLED;
if (disk_cache_)
disk_cache_->Disable();
}
void ServiceWorkerStorage::PurgeResources(const ResourceList& resources) {
if (!has_checked_for_stale_resources_)
DeleteStaleResources();
StartPurgingResources(resources);
}
ServiceWorkerStorage::ServiceWorkerStorage(
const base::FilePath& path,
base::WeakPtr<ServiceWorkerContextCore> context,
scoped_refptr<base::SequencedTaskRunner> database_task_runner,
storage::QuotaManagerProxy* quota_manager_proxy,
storage::SpecialStoragePolicy* special_storage_policy)
: next_registration_id_(blink::mojom::kInvalidServiceWorkerRegistrationId),
next_version_id_(blink::mojom::kInvalidServiceWorkerVersionId),
next_resource_id_(kInvalidServiceWorkerResourceId),
state_(UNINITIALIZED),
expecting_done_with_disk_on_disable_(false),
path_(path),
context_(context),
database_task_runner_(std::move(database_task_runner)),
quota_manager_proxy_(quota_manager_proxy),
special_storage_policy_(special_storage_policy),
is_purge_pending_(false),
has_checked_for_stale_resources_(false),
weak_factory_(this) {
DCHECK(context_);
database_.reset(new ServiceWorkerDatabase(GetDatabasePath()));
}
base::FilePath ServiceWorkerStorage::GetDatabasePath() {
if (path_.empty())
return base::FilePath();
return path_.Append(ServiceWorkerContextCore::kServiceWorkerDirectory)
.Append(kDatabaseName);
}
base::FilePath ServiceWorkerStorage::GetDiskCachePath() {
if (path_.empty())
return base::FilePath();
return path_.Append(ServiceWorkerContextCore::kServiceWorkerDirectory)
.Append(kDiskCacheName);
}
bool ServiceWorkerStorage::LazyInitializeForTest(base::OnceClosure callback) {
if (state_ == UNINITIALIZED || state_ == INITIALIZING) {
LazyInitialize(std::move(callback));
return false;
}
return !IsDisabled();
}
void ServiceWorkerStorage::LazyInitialize(base::OnceClosure callback) {
DCHECK(state_ == UNINITIALIZED || state_ == INITIALIZING) << state_;
pending_tasks_.push_back(std::move(callback));
if (state_ == INITIALIZING) {
return;
}
state_ = INITIALIZING;
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(&ReadInitialDataFromDB, database_.get(),
base::ThreadTaskRunnerHandle::Get(),
base::BindOnce(&ServiceWorkerStorage::DidReadInitialData,
weak_factory_.GetWeakPtr())));
}
void ServiceWorkerStorage::DidReadInitialData(
std::unique_ptr<InitialData> data,
ServiceWorkerDatabase::Status status) {
DCHECK(data);
DCHECK_EQ(INITIALIZING, state_);
if (status == ServiceWorkerDatabase::STATUS_OK) {
next_registration_id_ = data->next_registration_id;
next_version_id_ = data->next_version_id;
next_resource_id_ = data->next_resource_id;
registered_origins_.swap(data->origins);
state_ = INITIALIZED;
ServiceWorkerMetrics::RecordRegisteredOriginCount(
registered_origins_.size());
} else {
DVLOG(2) << "Failed to initialize: "
<< ServiceWorkerDatabase::StatusToString(status);
ScheduleDeleteAndStartOver();
}
for (base::OnceClosure& task : pending_tasks_)
RunSoon(FROM_HERE, std::move(task));
pending_tasks_.clear();
}
void ServiceWorkerStorage::DidFindRegistrationForDocument(
const GURL& document_url,
FindRegistrationCallback callback,
int64_t callback_id,
const ServiceWorkerDatabase::RegistrationData& data,
const ResourceList& resources,
ServiceWorkerDatabase::Status status) {
if (status == ServiceWorkerDatabase::STATUS_OK) {
ReturnFoundRegistration(std::move(callback), data, resources);
TRACE_EVENT_ASYNC_END1(
"ServiceWorker",
"ServiceWorkerStorage::FindRegistrationForDocument",
callback_id,
"Status", ServiceWorkerDatabase::StatusToString(status));
return;
}
if (status == ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND) {
// Look for something currently being installed.
scoped_refptr<ServiceWorkerRegistration> installing_registration =
FindInstallingRegistrationForDocument(document_url);
ServiceWorkerStatusCode installing_status =
installing_registration ? SERVICE_WORKER_OK
: SERVICE_WORKER_ERROR_NOT_FOUND;
std::move(callback).Run(installing_status,
std::move(installing_registration));
TRACE_EVENT_ASYNC_END2(
"ServiceWorker",
"ServiceWorkerStorage::FindRegistrationForDocument",
callback_id,
"Status", ServiceWorkerDatabase::StatusToString(status),
"Info",
(installing_status == SERVICE_WORKER_OK) ?
"Installing registration is found" :
"Any registrations are not found");
return;
}
ScheduleDeleteAndStartOver();
std::move(callback).Run(DatabaseStatusToStatusCode(status),
scoped_refptr<ServiceWorkerRegistration>());
TRACE_EVENT_ASYNC_END1(
"ServiceWorker",
"ServiceWorkerStorage::FindRegistrationForDocument",
callback_id,
"Status", ServiceWorkerDatabase::StatusToString(status));
}
void ServiceWorkerStorage::DidFindRegistrationForPattern(
const GURL& scope,
FindRegistrationCallback callback,
const ServiceWorkerDatabase::RegistrationData& data,
const ResourceList& resources,
ServiceWorkerDatabase::Status status) {
if (status == ServiceWorkerDatabase::STATUS_OK) {
ReturnFoundRegistration(std::move(callback), data, resources);
return;
}
if (status == ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND) {
scoped_refptr<ServiceWorkerRegistration> installing_registration =
FindInstallingRegistrationForPattern(scope);
ServiceWorkerStatusCode installing_status =
installing_registration ? SERVICE_WORKER_OK
: SERVICE_WORKER_ERROR_NOT_FOUND;
std::move(callback).Run(installing_status,
std::move(installing_registration));
return;
}
ScheduleDeleteAndStartOver();
std::move(callback).Run(DatabaseStatusToStatusCode(status),
scoped_refptr<ServiceWorkerRegistration>());
}
void ServiceWorkerStorage::DidFindRegistrationForId(
FindRegistrationCallback callback,
const ServiceWorkerDatabase::RegistrationData& data,
const ResourceList& resources,
ServiceWorkerDatabase::Status status) {
if (status == ServiceWorkerDatabase::STATUS_OK) {
ReturnFoundRegistration(std::move(callback), data, resources);
return;
}
if (status == ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND) {
// TODO(nhiroki): Find a registration in |installing_registrations_|.
std::move(callback).Run(DatabaseStatusToStatusCode(status),
scoped_refptr<ServiceWorkerRegistration>());
return;
}
ScheduleDeleteAndStartOver();
std::move(callback).Run(DatabaseStatusToStatusCode(status),
scoped_refptr<ServiceWorkerRegistration>());
}
void ServiceWorkerStorage::ReturnFoundRegistration(
FindRegistrationCallback callback,
const ServiceWorkerDatabase::RegistrationData& data,
const ResourceList& resources) {
DCHECK(!resources.empty());
scoped_refptr<ServiceWorkerRegistration> registration =
GetOrCreateRegistration(data, resources);
CompleteFindNow(std::move(registration), SERVICE_WORKER_OK,
std::move(callback));
}
void ServiceWorkerStorage::DidGetRegistrationsForOrigin(
GetRegistrationsCallback callback,
RegistrationList* registration_data_list,
std::vector<ResourceList>* resources_list,
const GURL& origin_filter,
ServiceWorkerDatabase::Status status) {
DCHECK(registration_data_list);
DCHECK(resources_list);
DCHECK(origin_filter.is_valid());
if (status != ServiceWorkerDatabase::STATUS_OK &&
status != ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND) {
ScheduleDeleteAndStartOver();
std::move(callback).Run(
DatabaseStatusToStatusCode(status),
std::vector<scoped_refptr<ServiceWorkerRegistration>>());
return;
}
// Add all stored registrations.
std::set<int64_t> registration_ids;
std::vector<scoped_refptr<ServiceWorkerRegistration>> registrations;
size_t index = 0;
for (const auto& registration_data : *registration_data_list) {
registration_ids.insert(registration_data.registration_id);
registrations.push_back(GetOrCreateRegistration(
registration_data, resources_list->at(index++)));
}
// Add unstored registrations that are being installed.
for (const auto& registration : installing_registrations_) {
if (registration.second->pattern().GetOrigin() != origin_filter)
continue;
if (registration_ids.insert(registration.first).second)
registrations.push_back(registration.second);
}
std::move(callback).Run(SERVICE_WORKER_OK, std::move(registrations));
}
void ServiceWorkerStorage::DidGetAllRegistrationsInfos(
GetRegistrationsInfosCallback callback,
RegistrationList* registration_data_list,
ServiceWorkerDatabase::Status status) {
DCHECK(registration_data_list);
if (status != ServiceWorkerDatabase::STATUS_OK &&
status != ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND) {
ScheduleDeleteAndStartOver();
std::move(callback).Run(DatabaseStatusToStatusCode(status),
std::vector<ServiceWorkerRegistrationInfo>());
return;
}
// Add all stored registrations.
std::set<int64_t> pushed_registrations;
std::vector<ServiceWorkerRegistrationInfo> infos;
for (const auto& registration_data : *registration_data_list) {
const bool inserted =
pushed_registrations.insert(registration_data.registration_id).second;
DCHECK(inserted);
ServiceWorkerRegistration* registration =
context_->GetLiveRegistration(registration_data.registration_id);
if (registration) {
infos.push_back(registration->GetInfo());
continue;
}
ServiceWorkerRegistrationInfo info;
info.pattern = registration_data.scope;
info.update_via_cache = registration_data.update_via_cache;
info.registration_id = registration_data.registration_id;
info.stored_version_size_bytes =
registration_data.resources_total_size_bytes;
if (ServiceWorkerVersion* version =
context_->GetLiveVersion(registration_data.version_id)) {
if (registration_data.is_active)
info.active_version = version->GetInfo();
else
info.waiting_version = version->GetInfo();
infos.push_back(info);
continue;
}
if (registration_data.is_active) {
info.active_version.status = ServiceWorkerVersion::ACTIVATED;
info.active_version.script_url = registration_data.script;
info.active_version.version_id = registration_data.version_id;
info.active_version.registration_id = registration_data.registration_id;
info.active_version.fetch_handler_existence =
registration_data.has_fetch_handler
? ServiceWorkerVersion::FetchHandlerExistence::EXISTS
: ServiceWorkerVersion::FetchHandlerExistence::DOES_NOT_EXIST;
} else {
info.waiting_version.status = ServiceWorkerVersion::INSTALLED;
info.waiting_version.script_url = registration_data.script;
info.waiting_version.version_id = registration_data.version_id;
info.waiting_version.registration_id = registration_data.registration_id;
info.waiting_version.fetch_handler_existence =
registration_data.has_fetch_handler
? ServiceWorkerVersion::FetchHandlerExistence::EXISTS
: ServiceWorkerVersion::FetchHandlerExistence::DOES_NOT_EXIST;
}
infos.push_back(info);
}
// Add unstored registrations that are being installed.
for (const auto& registration : installing_registrations_) {
if (pushed_registrations.insert(registration.first).second)
infos.push_back(registration.second->GetInfo());
}
std::move(callback).Run(SERVICE_WORKER_OK, infos);
}
void ServiceWorkerStorage::DidStoreRegistration(
StatusCallback callback,
const ServiceWorkerDatabase::RegistrationData& new_version,
const GURL& origin,
const ServiceWorkerDatabase::RegistrationData& deleted_version,
const std::vector<int64_t>& newly_purgeable_resources,
ServiceWorkerDatabase::Status status) {
if (status != ServiceWorkerDatabase::STATUS_OK) {
ScheduleDeleteAndStartOver();
std::move(callback).Run(DatabaseStatusToStatusCode(status));
return;
}
registered_origins_.insert(origin);
scoped_refptr<ServiceWorkerRegistration> registration =
context_->GetLiveRegistration(new_version.registration_id);
if (registration) {
registration->set_resources_total_size_bytes(
new_version.resources_total_size_bytes);
}
if (quota_manager_proxy_) {
// Can be nullptr in tests.
quota_manager_proxy_->NotifyStorageModified(
storage::QuotaClient::kServiceWorker, origin,
blink::mojom::StorageType::kTemporary,
new_version.resources_total_size_bytes -
deleted_version.resources_total_size_bytes);
}
std::move(callback).Run(SERVICE_WORKER_OK);
if (!context_->GetLiveVersion(deleted_version.version_id))
StartPurgingResources(newly_purgeable_resources);
}
void ServiceWorkerStorage::DidUpdateToActiveState(
StatusCallback callback,
ServiceWorkerDatabase::Status status) {
if (status != ServiceWorkerDatabase::STATUS_OK &&
status != ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND) {
ScheduleDeleteAndStartOver();
}
std::move(callback).Run(DatabaseStatusToStatusCode(status));
}
void ServiceWorkerStorage::DidDeleteRegistration(
std::unique_ptr<DidDeleteRegistrationParams> params,
OriginState origin_state,
const ServiceWorkerDatabase::RegistrationData& deleted_version,
const std::vector<int64_t>& newly_purgeable_resources,
ServiceWorkerDatabase::Status status) {
pending_deletions_.erase(params->registration_id);
if (status != ServiceWorkerDatabase::STATUS_OK) {
ScheduleDeleteAndStartOver();
std::move(params->callback).Run(DatabaseStatusToStatusCode(status));
return;
}
if (quota_manager_proxy_) {
// Can be nullptr in tests.
quota_manager_proxy_->NotifyStorageModified(
storage::QuotaClient::kServiceWorker, params->origin,
blink::mojom::StorageType::kTemporary,
-deleted_version.resources_total_size_bytes);
}
if (origin_state == OriginState::kDelete)
registered_origins_.erase(params->origin);
std::move(params->callback).Run(SERVICE_WORKER_OK);
if (!context_->GetLiveVersion(deleted_version.version_id))
StartPurgingResources(newly_purgeable_resources);
}
void ServiceWorkerStorage::DidWriteUncommittedResourceIds(
ServiceWorkerDatabase::Status status) {
if (status != ServiceWorkerDatabase::STATUS_OK)
ScheduleDeleteAndStartOver();
}
void ServiceWorkerStorage::DidPurgeUncommittedResourceIds(
const std::set<int64_t>& resource_ids,
ServiceWorkerDatabase::Status status) {
if (status != ServiceWorkerDatabase::STATUS_OK) {
ScheduleDeleteAndStartOver();
return;
}
StartPurgingResources(resource_ids);
}
void ServiceWorkerStorage::DidStoreUserData(
StatusCallback callback,
ServiceWorkerDatabase::Status status) {
// |status| can be NOT_FOUND when the associated registration did not exist in
// the database. In the case, we don't have to schedule the corruption
// recovery.
if (status != ServiceWorkerDatabase::STATUS_OK &&
status != ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND) {
ScheduleDeleteAndStartOver();
}
std::move(callback).Run(DatabaseStatusToStatusCode(status));
}
void ServiceWorkerStorage::DidGetUserData(
GetUserDataCallback callback,
const std::vector<std::string>& data,
ServiceWorkerDatabase::Status status) {
if (status != ServiceWorkerDatabase::STATUS_OK &&
status != ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND) {
ScheduleDeleteAndStartOver();
}
std::move(callback).Run(data, DatabaseStatusToStatusCode(status));
}
void ServiceWorkerStorage::DidGetUserKeysAndData(
GetUserKeysAndDataCallback callback,
const base::flat_map<std::string, std::string>& data_map,
ServiceWorkerDatabase::Status status) {
if (status != ServiceWorkerDatabase::STATUS_OK &&
status != ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND) {
ScheduleDeleteAndStartOver();
}
std::move(callback).Run(data_map, DatabaseStatusToStatusCode(status));
}
void ServiceWorkerStorage::DidDeleteUserData(
StatusCallback callback,
ServiceWorkerDatabase::Status status) {
if (status != ServiceWorkerDatabase::STATUS_OK)
ScheduleDeleteAndStartOver();
std::move(callback).Run(DatabaseStatusToStatusCode(status));
}
void ServiceWorkerStorage::DidGetUserDataForAllRegistrations(
GetUserDataForAllRegistrationsCallback callback,
const std::vector<std::pair<int64_t, std::string>>& user_data,
ServiceWorkerDatabase::Status status) {
if (status != ServiceWorkerDatabase::STATUS_OK)
ScheduleDeleteAndStartOver();
std::move(callback).Run(user_data, DatabaseStatusToStatusCode(status));
}
scoped_refptr<ServiceWorkerRegistration>
ServiceWorkerStorage::GetOrCreateRegistration(
const ServiceWorkerDatabase::RegistrationData& data,
const ResourceList& resources) {
scoped_refptr<ServiceWorkerRegistration> registration =
context_->GetLiveRegistration(data.registration_id);
if (registration)
return registration;
blink::mojom::ServiceWorkerRegistrationOptions options(data.scope,
data.update_via_cache);
registration =
new ServiceWorkerRegistration(options, data.registration_id, context_);
registration->set_resources_total_size_bytes(data.resources_total_size_bytes);
registration->set_last_update_check(data.last_update_check);
if (pending_deletions_.find(data.registration_id) !=
pending_deletions_.end()) {
registration->set_is_deleted(true);
}
scoped_refptr<ServiceWorkerVersion> version =
context_->GetLiveVersion(data.version_id);
if (!version) {
version = new ServiceWorkerVersion(
registration.get(), data.script, data.version_id, context_);
version->set_fetch_handler_existence(
data.has_fetch_handler
? ServiceWorkerVersion::FetchHandlerExistence::EXISTS
: ServiceWorkerVersion::FetchHandlerExistence::DOES_NOT_EXIST);
version->SetStatus(data.is_active ?
ServiceWorkerVersion::ACTIVATED : ServiceWorkerVersion::INSTALLED);
version->script_cache_map()->SetResources(resources);
if (data.origin_trial_tokens)
version->SetValidOriginTrialTokens(*data.origin_trial_tokens);
// Some features may be outside the valid range of features, if the data on
// disk was written by a later version of Chrome than currently running
// (crbug.com/758419).
// TODO(falken): Maybe Chrome should have a generic mechanism to detect
// profile downgrade and just abort? Or we could just crash here, but that
// seems extreme and difficult for a user to escape.
std::set<uint32_t> used_features = data.used_features;
for (auto it = used_features.begin(); it != used_features.end();) {
if (*it >=
static_cast<uint32_t>(blink::mojom::WebFeature::kNumberOfFeatures)) {
it = used_features.erase(it);
} else {
++it;
}
}
version->set_used_features(used_features);
}
if (version->status() == ServiceWorkerVersion::ACTIVATED)
registration->SetActiveVersion(version);
else if (version->status() == ServiceWorkerVersion::INSTALLED)
registration->SetWaitingVersion(version);
else
NOTREACHED();
registration->EnableNavigationPreload(data.navigation_preload_state.enabled);
registration->SetNavigationPreloadHeader(
data.navigation_preload_state.header);
return registration;
}
ServiceWorkerRegistration*
ServiceWorkerStorage::FindInstallingRegistrationForDocument(
const GURL& document_url) {
DCHECK(!document_url.has_ref());
LongestScopeMatcher matcher(document_url);
ServiceWorkerRegistration* match = nullptr;
// TODO(nhiroki): This searches over installing registrations linearly and it
// couldn't be scalable. Maybe the regs should be partitioned by origin.
for (const auto& registration : installing_registrations_)
if (matcher.MatchLongest(registration.second->pattern()))
match = registration.second.get();
return match;
}
ServiceWorkerRegistration*
ServiceWorkerStorage::FindInstallingRegistrationForPattern(const GURL& scope) {
for (const auto& registration : installing_registrations_)
if (registration.second->pattern() == scope)
return registration.second.get();
return nullptr;
}
ServiceWorkerRegistration*
ServiceWorkerStorage::FindInstallingRegistrationForId(int64_t registration_id) {
RegistrationRefsById::const_iterator found =
installing_registrations_.find(registration_id);
if (found == installing_registrations_.end())
return nullptr;
return found->second.get();
}
ServiceWorkerDiskCache* ServiceWorkerStorage::disk_cache() {
DCHECK(INITIALIZED == state_ || DISABLED == state_) << state_;
if (disk_cache_)
return disk_cache_.get();
disk_cache_.reset(new ServiceWorkerDiskCache);
if (IsDisabled()) {
disk_cache_->Disable();
return disk_cache_.get();
}
base::FilePath path = GetDiskCachePath();
if (path.empty()) {
int rv = disk_cache_->InitWithMemBackend(
kMaxServiceWorkerStorageMemDiskCacheSize, net::CompletionCallback());
DCHECK_EQ(net::OK, rv);
return disk_cache_.get();
}
InitializeDiskCache();
return disk_cache_.get();
}
void ServiceWorkerStorage::InitializeDiskCache() {
disk_cache_->set_is_waiting_to_initialize(false);
expecting_done_with_disk_on_disable_ = true;
int rv = disk_cache_->InitWithDiskBackend(
GetDiskCachePath(), kMaxServiceWorkerStorageDiskCacheSize, false,
base::BindOnce(&ServiceWorkerStorage::DiskCacheImplDoneWithDisk,
weak_factory_.GetWeakPtr()),
base::Bind(&ServiceWorkerStorage::OnDiskCacheInitialized,
weak_factory_.GetWeakPtr()));
if (rv != net::ERR_IO_PENDING)
OnDiskCacheInitialized(rv);
}
void ServiceWorkerStorage::OnDiskCacheInitialized(int rv) {
if (rv != net::OK) {
LOG(ERROR) << "Failed to open the serviceworker diskcache: "
<< net::ErrorToString(rv);
ScheduleDeleteAndStartOver();
}
ServiceWorkerMetrics::CountInitDiskCacheResult(rv == net::OK);
}
void ServiceWorkerStorage::StartPurgingResources(
const std::set<int64_t>& resource_ids) {
DCHECK(has_checked_for_stale_resources_);
for (int64_t resource_id : resource_ids)
purgeable_resource_ids_.push_back(resource_id);
ContinuePurgingResources();
}
void ServiceWorkerStorage::StartPurgingResources(
const std::vector<int64_t>& resource_ids) {
DCHECK(has_checked_for_stale_resources_);
for (int64_t resource_id : resource_ids)
purgeable_resource_ids_.push_back(resource_id);
ContinuePurgingResources();
}
void ServiceWorkerStorage::StartPurgingResources(
const ResourceList& resources) {
DCHECK(has_checked_for_stale_resources_);
for (const auto& resource : resources)
purgeable_resource_ids_.push_back(resource.resource_id);
ContinuePurgingResources();
}
void ServiceWorkerStorage::ContinuePurgingResources() {
if (purgeable_resource_ids_.empty() || is_purge_pending_)
return;
// Do one at a time until we're done, use RunSoon to avoid recursion when
// DoomEntry returns immediately.
is_purge_pending_ = true;
int64_t id = purgeable_resource_ids_.front();
purgeable_resource_ids_.pop_front();
RunSoon(FROM_HERE, base::BindOnce(&ServiceWorkerStorage::PurgeResource,
weak_factory_.GetWeakPtr(), id));
}
void ServiceWorkerStorage::PurgeResource(int64_t id) {
DCHECK(is_purge_pending_);
int rv = disk_cache()->DoomEntry(
id, base::Bind(&ServiceWorkerStorage::OnResourcePurged,
weak_factory_.GetWeakPtr(), id));
if (rv != net::ERR_IO_PENDING)
OnResourcePurged(id, rv);
}
void ServiceWorkerStorage::OnResourcePurged(int64_t id, int rv) {
DCHECK(is_purge_pending_);
is_purge_pending_ = false;
ServiceWorkerMetrics::RecordPurgeResourceResult(rv);
std::set<int64_t> ids = {id};
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
base::IgnoreResult(&ServiceWorkerDatabase::ClearPurgeableResourceIds),
base::Unretained(database_.get()), ids));
// Continue purging resources regardless of the previous result.
ContinuePurgingResources();
}
void ServiceWorkerStorage::DeleteStaleResources() {
DCHECK(!has_checked_for_stale_resources_);
has_checked_for_stale_resources_ = true;
database_task_runner_->PostTask(
FROM_HERE,
base::BindOnce(
&ServiceWorkerStorage::CollectStaleResourcesFromDB, database_.get(),
base::ThreadTaskRunnerHandle::Get(),
base::BindOnce(&ServiceWorkerStorage::DidCollectStaleResources,
weak_factory_.GetWeakPtr())));
}
void ServiceWorkerStorage::DidCollectStaleResources(
const std::vector<int64_t>& stale_resource_ids,
ServiceWorkerDatabase::Status status) {
if (status != ServiceWorkerDatabase::STATUS_OK) {
DCHECK_NE(ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND, status);
ScheduleDeleteAndStartOver();
return;
}
StartPurgingResources(stale_resource_ids);
}
void ServiceWorkerStorage::ClearSessionOnlyOrigins() {
// Can be null in tests.
if (!special_storage_policy_)
return;
if (!special_storage_policy_->HasSessionOnlyOrigins())
return;
std::set<GURL> session_only_origins;
for (const GURL& origin : registered_origins_) {
if (special_storage_policy_->IsStorageSessionOnly(origin))
session_only_origins.insert(origin);
}
database_task_runner_->PostTask(
FROM_HERE, base::BindOnce(&DeleteAllDataForOriginsFromDB, database_.get(),
session_only_origins));
}
// static
void ServiceWorkerStorage::CollectStaleResourcesFromDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
GetResourcesCallback callback) {
std::set<int64_t> ids;
ServiceWorkerDatabase::Status status =
database->GetUncommittedResourceIds(&ids);
if (status != ServiceWorkerDatabase::STATUS_OK) {
original_task_runner->PostTask(
FROM_HERE,
base::BindOnce(std::move(callback),
std::vector<int64_t>(ids.begin(), ids.end()), status));
return;
}
status = database->PurgeUncommittedResourceIds(ids);
if (status != ServiceWorkerDatabase::STATUS_OK) {
original_task_runner->PostTask(
FROM_HERE,
base::BindOnce(std::move(callback),
std::vector<int64_t>(ids.begin(), ids.end()), status));
return;
}
ids.clear();
status = database->GetPurgeableResourceIds(&ids);
original_task_runner->PostTask(
FROM_HERE,
base::BindOnce(std::move(callback),
std::vector<int64_t>(ids.begin(), ids.end()), status));
}
// static
void ServiceWorkerStorage::ReadInitialDataFromDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
InitializeCallback callback) {
DCHECK(database);
std::unique_ptr<ServiceWorkerStorage::InitialData> data(
new ServiceWorkerStorage::InitialData());
ServiceWorkerDatabase::Status status =
database->GetNextAvailableIds(&data->next_registration_id,
&data->next_version_id,
&data->next_resource_id);
if (status != ServiceWorkerDatabase::STATUS_OK) {
original_task_runner->PostTask(
FROM_HERE,
base::BindOnce(std::move(callback), std::move(data), status));
return;
}
status = database->GetOriginsWithRegistrations(&data->origins);
if (status != ServiceWorkerDatabase::STATUS_OK) {
original_task_runner->PostTask(
FROM_HERE,
base::BindOnce(std::move(callback), std::move(data), status));
return;
}
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback), std::move(data), status));
}
void ServiceWorkerStorage::DeleteRegistrationFromDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
int64_t registration_id,
const GURL& origin,
DeleteRegistrationCallback callback) {
DCHECK(database);
ServiceWorkerDatabase::RegistrationData deleted_version;
std::vector<int64_t> newly_purgeable_resources;
ServiceWorkerDatabase::Status status = database->DeleteRegistration(
registration_id, origin, &deleted_version, &newly_purgeable_resources);
if (status != ServiceWorkerDatabase::STATUS_OK) {
original_task_runner->PostTask(
FROM_HERE,
base::BindOnce(std::move(callback), OriginState::kKeep, deleted_version,
std::vector<int64_t>(), status));
return;
}
// TODO(nhiroki): Add convenient method to ServiceWorkerDatabase to check the
// unique origin list.
RegistrationList registrations;
status = database->GetRegistrationsForOrigin(origin, &registrations, nullptr);
if (status != ServiceWorkerDatabase::STATUS_OK) {
original_task_runner->PostTask(
FROM_HERE,
base::BindOnce(std::move(callback), OriginState::kKeep, deleted_version,
std::vector<int64_t>(), status));
return;
}
OriginState origin_state =
registrations.empty() ? OriginState::kDelete : OriginState::kKeep;
original_task_runner->PostTask(
FROM_HERE,
base::BindOnce(std::move(callback), origin_state, deleted_version,
newly_purgeable_resources, status));
}
void ServiceWorkerStorage::WriteRegistrationInDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
const ServiceWorkerDatabase::RegistrationData& data,
const ResourceList& resources,
WriteRegistrationCallback callback) {
DCHECK(database);
ServiceWorkerDatabase::RegistrationData deleted_version;
std::vector<int64_t> newly_purgeable_resources;
ServiceWorkerDatabase::Status status = database->WriteRegistration(
data, resources, &deleted_version, &newly_purgeable_resources);
original_task_runner->PostTask(
FROM_HERE,
base::BindOnce(std::move(callback), data.script.GetOrigin(),
deleted_version, newly_purgeable_resources, status));
}
// static
void ServiceWorkerStorage::FindForDocumentInDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
const GURL& document_url,
FindInDBCallback callback) {
GURL origin = document_url.GetOrigin();
RegistrationList registration_data_list;
ServiceWorkerDatabase::Status status = database->GetRegistrationsForOrigin(
origin, &registration_data_list, nullptr);
if (status != ServiceWorkerDatabase::STATUS_OK) {
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback),
ServiceWorkerDatabase::RegistrationData(),
ResourceList(), status));
return;
}
ServiceWorkerDatabase::RegistrationData data;
ResourceList resources;
status = ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND;
// Find one with a pattern match.
LongestScopeMatcher matcher(document_url);
int64_t match = blink::mojom::kInvalidServiceWorkerRegistrationId;
for (const auto& registration_data : registration_data_list)
if (matcher.MatchLongest(registration_data.scope))
match = registration_data.registration_id;
if (match != blink::mojom::kInvalidServiceWorkerRegistrationId)
status = database->ReadRegistration(match, origin, &data, &resources);
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback), data, resources, status));
}
// static
void ServiceWorkerStorage::FindForPatternInDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
const GURL& scope,
FindInDBCallback callback) {
GURL origin = scope.GetOrigin();
RegistrationList registration_data_list;
ServiceWorkerDatabase::Status status = database->GetRegistrationsForOrigin(
origin, &registration_data_list, nullptr);
if (status != ServiceWorkerDatabase::STATUS_OK) {
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback),
ServiceWorkerDatabase::RegistrationData(),
ResourceList(), status));
return;
}
// Find one with an exact matching scope.
ServiceWorkerDatabase::RegistrationData data;
ResourceList resources;
status = ServiceWorkerDatabase::STATUS_ERROR_NOT_FOUND;
for (const auto& registration_data : registration_data_list) {
if (scope != registration_data.scope)
continue;
status = database->ReadRegistration(registration_data.registration_id,
origin, &data, &resources);
break; // We're done looping.
}
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback), data, resources, status));
}
// static
void ServiceWorkerStorage::FindForIdInDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
int64_t registration_id,
const GURL& origin,
FindInDBCallback callback) {
ServiceWorkerDatabase::RegistrationData data;
ResourceList resources;
ServiceWorkerDatabase::Status status =
database->ReadRegistration(registration_id, origin, &data, &resources);
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback), data, resources, status));
}
// static
void ServiceWorkerStorage::FindForIdOnlyInDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
int64_t registration_id,
FindInDBCallback callback) {
GURL origin;
ServiceWorkerDatabase::Status status =
database->ReadRegistrationOrigin(registration_id, &origin);
if (status != ServiceWorkerDatabase::STATUS_OK) {
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback),
ServiceWorkerDatabase::RegistrationData(),
ResourceList(), status));
return;
}
FindForIdInDB(database, original_task_runner, registration_id, origin,
std::move(callback));
}
void ServiceWorkerStorage::GetUserDataInDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
int64_t registration_id,
const std::vector<std::string>& keys,
GetUserDataInDBCallback callback) {
std::vector<std::string> values;
ServiceWorkerDatabase::Status status =
database->ReadUserData(registration_id, keys, &values);
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback), values, status));
}
void ServiceWorkerStorage::GetUserDataByKeyPrefixInDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
int64_t registration_id,
const std::string& key_prefix,
GetUserDataInDBCallback callback) {
std::vector<std::string> values;
ServiceWorkerDatabase::Status status =
database->ReadUserDataByKeyPrefix(registration_id, key_prefix, &values);
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback), values, status));
}
void ServiceWorkerStorage::GetUserKeysAndDataByKeyPrefixInDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
int64_t registration_id,
const std::string& key_prefix,
GetUserKeysAndDataInDBCallback callback) {
base::flat_map<std::string, std::string> data_map;
ServiceWorkerDatabase::Status status =
database->ReadUserKeysAndDataByKeyPrefix(registration_id, key_prefix,
&data_map);
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback), data_map, status));
}
void ServiceWorkerStorage::GetUserDataForAllRegistrationsInDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
const std::string& key,
GetUserDataForAllRegistrationsInDBCallback callback) {
std::vector<std::pair<int64_t, std::string>> user_data;
ServiceWorkerDatabase::Status status =
database->ReadUserDataForAllRegistrations(key, &user_data);
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback), user_data, status));
}
void ServiceWorkerStorage::GetUserDataForAllRegistrationsByKeyPrefixInDB(
ServiceWorkerDatabase* database,
scoped_refptr<base::SequencedTaskRunner> original_task_runner,
const std::string& key_prefix,
GetUserDataForAllRegistrationsInDBCallback callback) {
std::vector<std::pair<int64_t, std::string>> user_data;
ServiceWorkerDatabase::Status status =
database->ReadUserDataForAllRegistrationsByKeyPrefix(key_prefix,
&user_data);
original_task_runner->PostTask(
FROM_HERE, base::BindOnce(std::move(callback), user_data, status));
}
void ServiceWorkerStorage::DeleteAllDataForOriginsFromDB(
ServiceWorkerDatabase* database,
const std::set<GURL>& origins) {
DCHECK(database);
std::vector<int64_t> newly_purgeable_resources;
database->DeleteAllDataForOrigins(origins, &newly_purgeable_resources);
}
bool ServiceWorkerStorage::IsDisabled() const {
return state_ == DISABLED;
}
// TODO(nhiroki): The corruption recovery should not be scheduled if the error
// is transient and it can get healed soon (e.g. IO error). To do that, the
// database should not disable itself when an error occurs and the storage
// controls it instead.
void ServiceWorkerStorage::ScheduleDeleteAndStartOver() {
// TODO(dmurph): Notify the quota manager somehow that all of our data is now
// removed.
if (state_ == DISABLED) {
// Recovery process has already been scheduled.
return;
}
Disable();
DVLOG(1) << "Schedule to delete the context and start over.";
context_->ScheduleDeleteAndStartOver();
}
void ServiceWorkerStorage::DidDeleteDatabase(
StatusCallback callback,
ServiceWorkerDatabase::Status status) {
DCHECK_EQ(DISABLED, state_);
if (status != ServiceWorkerDatabase::STATUS_OK) {
// Give up the corruption recovery until the browser restarts.
LOG(ERROR) << "Failed to delete the database: "
<< ServiceWorkerDatabase::StatusToString(status);
ServiceWorkerMetrics::RecordDeleteAndStartOverResult(
ServiceWorkerMetrics::DELETE_DATABASE_ERROR);
std::move(callback).Run(DatabaseStatusToStatusCode(status));
return;
}
DVLOG(1) << "Deleted ServiceWorkerDatabase successfully.";
// Delete the disk cache. Use BLOCK_SHUTDOWN to try to avoid things being
// half-deleted.
// TODO(falken): Investigate if BLOCK_SHUTDOWN is needed, as the next startup
// is expected to cleanup the disk cache anyway. Also investigate whether
// ClearSessionOnlyOrigins() should try to delete relevant entries from the
// disk cache before shutdown.
// TODO(nhiroki): What if there is a bunch of files in the cache directory?
// Deleting the directory could take a long time and restart could be delayed.
// We should probably rename the directory and delete it later.
PostTaskWithTraitsAndReplyWithResult(
FROM_HERE, {base::MayBlock(), base::TaskShutdownBehavior::BLOCK_SHUTDOWN},
base::BindOnce(&base::DeleteFile, GetDiskCachePath(), true),
base::BindOnce(&ServiceWorkerStorage::DidDeleteDiskCache,
weak_factory_.GetWeakPtr(), std::move(callback)));
}
void ServiceWorkerStorage::DidDeleteDiskCache(StatusCallback callback,
bool result) {
DCHECK_EQ(DISABLED, state_);
if (!result) {
// Give up the corruption recovery until the browser restarts.
LOG(ERROR) << "Failed to delete the diskcache.";
ServiceWorkerMetrics::RecordDeleteAndStartOverResult(
ServiceWorkerMetrics::DELETE_DISK_CACHE_ERROR);
std::move(callback).Run(SERVICE_WORKER_ERROR_FAILED);
return;
}
DVLOG(1) << "Deleted ServiceWorkerDiskCache successfully.";
ServiceWorkerMetrics::RecordDeleteAndStartOverResult(
ServiceWorkerMetrics::DELETE_OK);
std::move(callback).Run(SERVICE_WORKER_OK);
}
} // namespace content