blob: 50ef46a53d6d8d329e34a071a6b93031abf4f4a8 [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/indexed_db/indexed_db_pre_close_task_queue.h"
#include "base/bind.h"
#include "base/metrics/histogram_macros.h"
#include "base/threading/sequenced_task_runner_handle.h"
#include "third_party/blink/public/common/indexeddb/indexeddb_metadata.h"
#include "third_party/leveldatabase/env_chromium.h"
using blink::IndexedDBDatabaseMetadata;
namespace content {
IndexedDBPreCloseTaskQueue::PreCloseTask::~PreCloseTask() {}
IndexedDBPreCloseTaskQueue::IndexedDBPreCloseTaskQueue(
std::list<std::unique_ptr<IndexedDBPreCloseTaskQueue::PreCloseTask>> tasks,
base::OnceClosure on_complete,
base::TimeDelta max_run_time,
std::unique_ptr<base::OneShotTimer> timer)
: tasks_(std::move(tasks)),
on_done_(std::move(on_complete)),
timeout_time_(max_run_time),
timeout_timer_(std::move(timer)),
task_runner_(base::SequencedTaskRunnerHandle::Get()),
ptr_factory_(this) {}
IndexedDBPreCloseTaskQueue::~IndexedDBPreCloseTaskQueue() {}
void IndexedDBPreCloseTaskQueue::StopForNewConnection() {
if (!started_ || done_)
return;
DCHECK(!tasks_.empty());
while (!tasks_.empty()) {
tasks_.front()->Stop(StopReason::NEW_CONNECTION);
tasks_.pop_front();
}
OnComplete();
}
void IndexedDBPreCloseTaskQueue::Start(
base::OnceCallback<leveldb::Status(std::vector<IndexedDBDatabaseMetadata>*)>
metadata_fetcher) {
DCHECK(!started_);
started_ = true;
if (tasks_.empty()) {
OnComplete();
return;
}
timeout_timer_->Start(
FROM_HERE, timeout_time_,
base::BindOnce(&IndexedDBPreCloseTaskQueue::StopForTimout,
ptr_factory_.GetWeakPtr()));
leveldb::Status status = std::move(metadata_fetcher).Run(&metadata_);
if (!status.ok()) {
StopForMetadataError(status);
return;
}
tasks_.front()->SetMetadata(&metadata_);
task_runner_->PostTask(FROM_HERE,
base::BindOnce(&IndexedDBPreCloseTaskQueue::RunLoop,
ptr_factory_.GetWeakPtr()));
}
void IndexedDBPreCloseTaskQueue::OnComplete() {
DCHECK(started_);
DCHECK(!done_);
ptr_factory_.InvalidateWeakPtrs();
timeout_timer_->Stop();
done_ = true;
std::move(on_done_).Run();
}
void IndexedDBPreCloseTaskQueue::StopForTimout() {
DCHECK(started_);
if (done_)
return;
while (!tasks_.empty()) {
tasks_.front()->Stop(StopReason::TIMEOUT);
tasks_.pop_front();
}
OnComplete();
}
void IndexedDBPreCloseTaskQueue::StopForMetadataError(
const leveldb::Status& status) {
if (done_)
return;
LOCAL_HISTOGRAM_ENUMERATION(
"WebCore.IndexedDB.IndexedDBPreCloseTaskList.MetadataError",
leveldb_env::GetLevelDBStatusUMAValue(status),
leveldb_env::LEVELDB_STATUS_MAX);
while (!tasks_.empty()) {
tasks_.front()->Stop(StopReason::METADATA_ERROR);
tasks_.pop_front();
}
OnComplete();
}
void IndexedDBPreCloseTaskQueue::RunLoop() {
if (done_)
return;
if (tasks_.empty()) {
OnComplete();
return;
}
bool done = tasks_.front()->RunRound();
if (done) {
tasks_.pop_front();
if (tasks_.empty()) {
OnComplete();
return;
}
tasks_.front()->SetMetadata(&metadata_);
}
task_runner_->PostTask(FROM_HERE,
base::BindOnce(&IndexedDBPreCloseTaskQueue::RunLoop,
ptr_factory_.GetWeakPtr()));
}
} // namespace content