blob: db0fd2686467dff5c1af8ca0c013874103de2b7b [file] [log] [blame]
// Copyright 2021 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/reporting/client/report_queue_factory.h"
#include <utility>
#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/strings/string_piece.h"
#include "base/task/bind_post_task.h"
#include "base/task/sequenced_task_runner.h"
#include "base/task/thread_pool.h"
#include "components/reporting/client/report_queue_configuration.h"
#include "components/reporting/client/report_queue_provider.h"
#include "components/reporting/util/backoff_settings.h"
#include "net/base/backoff_entry.h"
#define LOG_WITH_STATUS(LEVEL, MESSAGE, STATUS) \
VLOG(LEVEL) << MESSAGE << " status=" << STATUS.status();
namespace reporting {
// static
void ReportQueueFactory::Create(EventType event_type,
Destination destination,
SuccessCallback success_cb,
int64_t reserved_space) {
DCHECK(base::SequencedTaskRunner::HasCurrentDefault());
auto config_result = ReportQueueConfiguration::Create(
event_type, destination,
base::BindRepeating([]() { return Status::StatusOK(); }), reserved_space);
if (!config_result.ok()) {
LOG_WITH_STATUS(1, "ReportQueueConfiguration is invalid.", config_result);
return;
}
// Asynchronously create and try to set ReportQueue.
auto try_set_cb = CreateTrySetCallback(destination, std::move(success_cb),
GetBackoffEntry());
base::ThreadPool::PostTask(
FROM_HERE, base::BindOnce(ReportQueueProvider::CreateQueue,
std::move(config_result.ValueOrDie()),
std::move(try_set_cb)));
}
// static
std::unique_ptr<ReportQueue, base::OnTaskRunnerDeleter>
ReportQueueFactory::CreateSpeculativeReportQueue(EventType event_type,
Destination destination,
int64_t reserved_space) {
DCHECK(base::SequencedTaskRunner::HasCurrentDefault());
auto config_result = ReportQueueConfiguration::Create(
event_type, destination,
base::BindRepeating([]() { return Status::StatusOK(); }), reserved_space);
if (!config_result.ok()) {
DVLOG(1)
<< "Cannot initialize report queue. Invalid ReportQueueConfiguration: "
<< config_result.status();
return std::unique_ptr<ReportQueue, base::OnTaskRunnerDeleter>(
nullptr, base::OnTaskRunnerDeleter(
base::SequencedTaskRunner::GetCurrentDefault()));
}
auto speculative_queue_result = ReportQueueProvider::CreateSpeculativeQueue(
std::move(config_result.ValueOrDie()));
if (!speculative_queue_result.ok()) {
DVLOG(1) << "Failed to create speculative queue: "
<< speculative_queue_result.status();
return std::unique_ptr<ReportQueue, base::OnTaskRunnerDeleter>(
nullptr, base::OnTaskRunnerDeleter(
base::SequencedTaskRunner::GetCurrentDefault()));
}
return std::move(speculative_queue_result.ValueOrDie());
}
ReportQueueFactory::TrySetReportQueueCallback
ReportQueueFactory::CreateTrySetCallback(
Destination destination,
SuccessCallback success_cb,
std::unique_ptr<::net::BackoffEntry> backoff_entry) {
return base::BindPostTaskToCurrentDefault(base::BindOnce(
&ReportQueueFactory::TrySetReportQueue, std::move(success_cb)));
}
// static
void ReportQueueFactory::TrySetReportQueue(
SuccessCallback success_cb,
StatusOr<std::unique_ptr<ReportQueue>> report_queue_result) {
if (!report_queue_result.ok()) {
LOG_WITH_STATUS(1, "ReportQueue could not be created.",
report_queue_result);
return;
}
std::move(success_cb).Run(std::move(report_queue_result.ValueOrDie()));
}
} // namespace reporting