#include "media/capabilities/pending_operations.h"
#include <string>
#include "base/functional/bind.h"
#include "base/logging.h"
#include "base/metrics/histogram_functions.h"
#include "base/not_fatal_until.h"
#include "base/task/single_thread_task_runner.h"
namespace media {
namespace {
static constexpr base::TimeDelta kPendingOpTimeout = base::Seconds(30);
}
PendingOperations::PendingOperation::PendingOperation(
std::string uma_str,
std::unique_ptr<base::CancelableOnceClosure> timeout_closure)
: uma_str_(std::move(uma_str)),
timeout_closure_(std::move(timeout_closure)),
start_ticks_(base::TimeTicks::Now()) {
DVLOG(3) << __func__ << " Started " << uma_str_;
}
PendingOperations::PendingOperation::~PendingOperation() {
if (timeout_closure_ && !timeout_closure_->IsCancelled()) {
base::TimeDelta op_duration = base::TimeTicks::Now() - start_ticks_;
UmaHistogramOpTime(op_duration);
DVLOG(3) << __func__ << " Completed " << uma_str_ << " ("
<< op_duration.InMilliseconds() << ")";
timeout_closure_->Cancel();
}
}
void PendingOperations::PendingOperation::UmaHistogramOpTime(
base::TimeDelta duration) {
base::UmaHistogramCustomMicrosecondsTimes(
uma_str_, duration, base::Milliseconds(1), kPendingOpTimeout, 50);
}
void PendingOperations::PendingOperation::OnTimeout() {
UmaHistogramOpTime(kPendingOpTimeout);
LOG(WARNING) << " Timeout performing " << uma_str_
<< " operation on WebrtcVideoStatsDB";
timeout_closure_->Cancel();
}
PendingOperations::PendingOperations(std::string uma_prefix)
: uma_prefix_(std::move(uma_prefix)) {}
PendingOperations::~PendingOperations() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
}
PendingOperations::Id PendingOperations::Start(std::string_view uma_str) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
Id op_id = next_op_id_++;
auto timeout_closure = std::make_unique<base::CancelableOnceClosure>(
base::BindOnce(&PendingOperations::OnTimeout,
weak_ptr_factory_.GetWeakPtr(), op_id));
base::SingleThreadTaskRunner::GetCurrentDefault()->PostDelayedTask(
FROM_HERE, timeout_closure->callback(), kPendingOpTimeout);
pending_ops_.emplace(op_id, std::make_unique<PendingOperation>(
uma_prefix_ + std::string(uma_str),
std::move(timeout_closure)));
return op_id;
}
void PendingOperations::Complete(Id op_id) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
int count = pending_ops_.erase(op_id);
if (!count)
DVLOG(2) << __func__ << " DB operation completed after timeout.";
}
void PendingOperations::OnTimeout(Id op_id) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
auto it = pending_ops_.find(op_id);
CHECK(it != pending_ops_.end());
it->second->OnTimeout();
pending_ops_.erase(it);
}
}