blob: 09701f99dce2f5a69382b5d3b5b84f8dff2051a7 [file] [log] [blame]
// Copyright (c) 2012 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 "storage/browser/database/database_quota_client.h"
#include <stdint.h>
#include <memory>
#include <utility>
#include <vector>
#include "base/bind.h"
#include "base/bind_helpers.h"
#include "base/callback_helpers.h"
#include "base/location.h"
#include "base/task_runner_util.h"
#include "base/threading/sequenced_task_runner_handle.h"
#include "net/base/completion_once_callback.h"
#include "net/base/net_errors.h"
#include "net/base/url_util.h"
#include "storage/browser/database/database_tracker.h"
#include "storage/browser/database/database_util.h"
#include "storage/common/database/database_identifier.h"
#include "third_party/blink/public/mojom/quota/quota_types.mojom.h"
using blink::mojom::StorageType;
using storage::QuotaClient;
namespace storage {
namespace {
int64_t GetOriginUsageOnDBThread(DatabaseTracker* db_tracker,
const url::Origin& origin) {
OriginInfo info;
if (db_tracker->GetOriginInfo(storage::GetIdentifierFromOrigin(origin),
&info))
return info.TotalSize();
return 0;
}
void GetOriginsOnDBThread(DatabaseTracker* db_tracker,
std::set<url::Origin>* origins_ptr) {
std::vector<std::string> origin_identifiers;
if (db_tracker->GetAllOriginIdentifiers(&origin_identifiers)) {
for (const auto& identifier : origin_identifiers) {
origins_ptr->insert(storage::GetOriginFromIdentifier(identifier));
}
}
}
void GetOriginsForHostOnDBThread(DatabaseTracker* db_tracker,
std::set<url::Origin>* origins_ptr,
const std::string& host) {
std::vector<std::string> origin_identifiers;
if (db_tracker->GetAllOriginIdentifiers(&origin_identifiers)) {
for (const auto& identifier : origin_identifiers) {
url::Origin origin = storage::GetOriginFromIdentifier(identifier);
if (host == net::GetHostOrSpecFromURL(origin.GetURL()))
origins_ptr->insert(origin);
}
}
}
void DidGetQuotaClientOrigins(QuotaClient::GetOriginsCallback callback,
std::set<url::Origin>* origins_ptr) {
std::move(callback).Run(*origins_ptr);
}
void DidDeleteOriginData(base::SequencedTaskRunner* original_task_runner,
QuotaClient::DeletionCallback callback,
int result) {
if (result == net::ERR_IO_PENDING) {
// The callback will be invoked via
// DatabaseTracker::ScheduleDatabasesForDeletion.
return;
}
blink::mojom::QuotaStatusCode status;
if (result == net::OK)
status = blink::mojom::QuotaStatusCode::kOk;
else
status = blink::mojom::QuotaStatusCode::kUnknown;
original_task_runner->PostTask(FROM_HERE,
base::BindOnce(std::move(callback), status));
}
} // namespace
DatabaseQuotaClient::DatabaseQuotaClient(
scoped_refptr<DatabaseTracker> db_tracker)
: db_tracker_(std::move(db_tracker)) {}
DatabaseQuotaClient::~DatabaseQuotaClient() {
if (!db_tracker_->task_runner()->RunsTasksInCurrentSequence()) {
db_tracker_->task_runner()->ReleaseSoon(FROM_HERE, std::move(db_tracker_));
}
}
QuotaClient::ID DatabaseQuotaClient::id() const {
return kDatabase;
}
void DatabaseQuotaClient::OnQuotaManagerDestroyed() {
delete this;
}
void DatabaseQuotaClient::GetOriginUsage(const url::Origin& origin,
StorageType type,
GetUsageCallback callback) {
DCHECK(!callback.is_null());
DCHECK(db_tracker_.get());
// All databases are in the temp namespace for now.
if (type != StorageType::kTemporary) {
std::move(callback).Run(0);
return;
}
base::PostTaskAndReplyWithResult(
db_tracker_->task_runner(), FROM_HERE,
base::BindOnce(&GetOriginUsageOnDBThread, base::RetainedRef(db_tracker_),
origin),
std::move(callback));
}
void DatabaseQuotaClient::GetOriginsForType(StorageType type,
GetOriginsCallback callback) {
DCHECK(!callback.is_null());
DCHECK(db_tracker_.get());
// All databases are in the temp namespace for now.
if (type != StorageType::kTemporary) {
std::move(callback).Run(std::set<url::Origin>());
return;
}
std::set<url::Origin>* origins_ptr = new std::set<url::Origin>();
db_tracker_->task_runner()->PostTaskAndReply(
FROM_HERE,
base::BindOnce(&GetOriginsOnDBThread, base::RetainedRef(db_tracker_),
base::Unretained(origins_ptr)),
base::BindOnce(&DidGetQuotaClientOrigins, std::move(callback),
base::Owned(origins_ptr)));
}
void DatabaseQuotaClient::GetOriginsForHost(StorageType type,
const std::string& host,
GetOriginsCallback callback) {
DCHECK(!callback.is_null());
DCHECK(db_tracker_.get());
// All databases are in the temp namespace for now.
if (type != StorageType::kTemporary) {
std::move(callback).Run(std::set<url::Origin>());
return;
}
std::set<url::Origin>* origins_ptr = new std::set<url::Origin>();
db_tracker_->task_runner()->PostTaskAndReply(
FROM_HERE,
base::BindOnce(&GetOriginsForHostOnDBThread,
base::RetainedRef(db_tracker_),
base::Unretained(origins_ptr), host),
base::BindOnce(&DidGetQuotaClientOrigins, std::move(callback),
base::Owned(origins_ptr)));
}
void DatabaseQuotaClient::DeleteOriginData(const url::Origin& origin,
StorageType type,
DeletionCallback callback) {
DCHECK(!callback.is_null());
DCHECK(db_tracker_.get());
// All databases are in the temp namespace for now, so nothing to delete.
if (type != StorageType::kTemporary) {
std::move(callback).Run(blink::mojom::QuotaStatusCode::kOk);
return;
}
// DidDeleteOriginData() translates the net::Error response to a
// blink::mojom::QuotaStatusCode if necessary, and no-ops as appropriate if
// DatabaseTracker::ScheduleDatabasesForDeletion will also invoke the
// callback.
auto delete_callback = base::BindRepeating(
&DidDeleteOriginData,
base::RetainedRef(base::SequencedTaskRunnerHandle::Get()),
base::AdaptCallbackForRepeating(std::move(callback)));
base::PostTaskAndReplyWithResult(
db_tracker_->task_runner(), FROM_HERE,
base::BindOnce(&DatabaseTracker::DeleteDataForOrigin, db_tracker_, origin,
delete_callback),
net::CompletionOnceCallback(delete_callback));
}
bool DatabaseQuotaClient::DoesSupport(StorageType type) const {
return type == StorageType::kTemporary;
}
} // namespace storage