blob: 0029afb13186473ddf48ac6db13444822238aa3f [file] [log] [blame]
// Copyright 2017 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/background_fetch/storage/database_task.h"
#include <utility>
#include "base/metrics/histogram_functions.h"
#include "content/browser/background_fetch/background_fetch_data_manager.h"
#include "content/browser/background_fetch/background_fetch_data_manager_observer.h"
#include "content/browser/background_fetch/storage/database_helpers.h"
#include "content/browser/service_worker/service_worker_context_wrapper.h"
#include "content/public/browser/browser_thread.h"
#include "storage/browser/quota/quota_manager_proxy.h"
namespace content {
namespace background_fetch {
namespace {
void DidGetUsageAndQuota(DatabaseTask::IsQuotaAvailableCallback callback,
int64_t size,
blink::mojom::QuotaStatusCode status,
int64_t usage,
int64_t quota) {
bool is_available =
status == blink::mojom::QuotaStatusCode::kOk && (usage + size) <= quota;
std::move(callback).Run(is_available);
}
} // namespace
DatabaseTaskHost::DatabaseTaskHost() : weak_factory_(this) {}
DatabaseTaskHost::~DatabaseTaskHost() = default;
base::WeakPtr<DatabaseTaskHost> DatabaseTaskHost::GetWeakPtr() {
return weak_factory_.GetWeakPtr();
}
DatabaseTask::DatabaseTask(DatabaseTaskHost* host) : host_(host) {
DCHECK(host_);
// Hold a reference to the CacheStorageManager.
cache_manager_ = data_manager()->cache_manager();
}
DatabaseTask::~DatabaseTask() = default;
void DatabaseTask::Finished() {
DCHECK_CURRENTLY_ON(BrowserThread::IO);
// Post the OnTaskFinished callback to the same thread, to allow the the
// DatabaseTask to finish execution before deallocating it.
base::ThreadTaskRunnerHandle::Get()->PostTask(
FROM_HERE, base::BindOnce(&DatabaseTaskHost::OnTaskFinished,
host_->GetWeakPtr(), this));
}
void DatabaseTask::OnTaskFinished(DatabaseTask* finished_subtask) {
size_t erased = active_subtasks_.erase(finished_subtask);
DCHECK_EQ(erased, 1u);
}
void DatabaseTask::AddDatabaseTask(std::unique_ptr<DatabaseTask> task) {
DCHECK_EQ(task->host_, data_manager());
data_manager()->AddDatabaseTask(std::move(task));
}
void DatabaseTask::AddSubTask(std::unique_ptr<DatabaseTask> task) {
DCHECK_EQ(task->host_, this);
auto insert_result = active_subtasks_.emplace(task.get(), std::move(task));
insert_result.first->second->Start(); // Start the subtask.
}
void DatabaseTask::AbandonFetches(int64_t service_worker_registration_id) {
for (auto& observer : data_manager()->observers())
observer.OnServiceWorkerDatabaseCorrupted(service_worker_registration_id);
}
void DatabaseTask::IsQuotaAvailable(const url::Origin& origin,
int64_t size,
IsQuotaAvailableCallback callback) {
DCHECK(quota_manager_proxy());
DCHECK_GT(size, 0);
quota_manager_proxy()->GetUsageAndQuota(
base::ThreadTaskRunnerHandle::Get().get(), origin,
blink::mojom::StorageType::kTemporary,
base::BindOnce(&DidGetUsageAndQuota, std::move(callback), size));
}
void DatabaseTask::SetStorageError(BackgroundFetchStorageError error) {
DCHECK_NE(BackgroundFetchStorageError::kNone, error);
switch (storage_error_) {
case BackgroundFetchStorageError::kNone:
storage_error_ = error;
break;
case BackgroundFetchStorageError::kServiceWorkerStorageError:
case BackgroundFetchStorageError::kCacheStorageError:
DCHECK(error == BackgroundFetchStorageError::kServiceWorkerStorageError ||
error == BackgroundFetchStorageError::kCacheStorageError);
if (storage_error_ != error)
storage_error_ = BackgroundFetchStorageError::kStorageError;
break;
case BackgroundFetchStorageError::kStorageError:
break;
}
}
void DatabaseTask::SetStorageErrorAndFinish(BackgroundFetchStorageError error) {
SetStorageError(error);
FinishWithError(blink::mojom::BackgroundFetchError::STORAGE_ERROR);
}
void DatabaseTask::ReportStorageError() {
if (host_ != data_manager())
return; // This is a SubTask.
base::UmaHistogramEnumeration("BackgroundFetch.Storage." + HistogramName(),
storage_error_);
}
bool DatabaseTask::HasStorageError() {
return storage_error_ != BackgroundFetchStorageError::kNone;
}
std::string DatabaseTask::HistogramName() const {
NOTREACHED() << "HistogramName needs to be provided.";
return "GeneralDatabaseTask";
}
ServiceWorkerContextWrapper* DatabaseTask::service_worker_context() {
DCHECK(data_manager()->service_worker_context());
return data_manager()->service_worker_context();
}
CacheStorageManager* DatabaseTask::cache_manager() {
DCHECK(cache_manager_);
return cache_manager_.get();
}
std::set<std::string>& DatabaseTask::ref_counted_unique_ids() {
return data_manager()->ref_counted_unique_ids();
}
ChromeBlobStorageContext* DatabaseTask::blob_storage_context() {
return data_manager()->blob_storage_context();
}
BackgroundFetchDataManager* DatabaseTask::data_manager() {
return host_->data_manager();
}
storage::QuotaManagerProxy* DatabaseTask::quota_manager_proxy() {
return data_manager()->quota_manager_proxy();
}
} // namespace background_fetch
} // namespace content