blob: 9272533a20f4829cd446ff2d2eac8986fa4465aa [file] [log] [blame]
// Copyright 2022 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/metrics/periodic_collector.h"
#include <memory>
#include <optional>
#include <string>
#include <utility>
#include "base/check.h"
#include "base/functional/bind.h"
#include "base/location.h"
#include "base/metrics/histogram_functions.h"
#include "base/sequence_checker.h"
#include "base/task/sequenced_task_runner.h"
#include "base/time/time.h"
#include "components/reporting/client/report_queue.h"
#include "components/reporting/metrics/collector_base.h"
#include "components/reporting/metrics/metric_rate_controller.h"
#include "components/reporting/metrics/metric_report_queue.h"
#include "components/reporting/metrics/metric_reporting_controller.h"
#include "components/reporting/metrics/sampler.h"
#include "components/reporting/proto/synced/record_constants.pb.h"
namespace reporting {
PeriodicCollector::PeriodicCollector(Sampler* sampler,
MetricReportQueue* metric_report_queue,
ReportingSettings* reporting_settings,
const std::string& enable_setting_path,
bool setting_enabled_default_value,
const std::string& rate_setting_path,
base::TimeDelta default_rate,
int rate_unit_to_ms,
base::TimeDelta init_delay)
: CollectorBase(sampler),
metric_report_queue_(metric_report_queue),
rate_controller_(std::make_unique<MetricRateController>(
base::BindRepeating(&PeriodicCollector::Collect,
base::Unretained(this),
/*is_event_driven=*/false),
reporting_settings,
rate_setting_path,
default_rate,
rate_unit_to_ms)),
reporting_controller_(std::make_unique<MetricReportingController>(
reporting_settings,
enable_setting_path,
setting_enabled_default_value)) {
if (init_delay.is_zero()) {
SetReportingControllerCb();
return;
}
CHECK(base::SequencedTaskRunner::HasCurrentDefault());
base::SequencedTaskRunner::GetCurrentDefault()->PostDelayedTask(
FROM_HERE,
base::BindOnce(&PeriodicCollector::SetReportingControllerCb,
weak_ptr_factory_.GetWeakPtr()),
init_delay);
}
PeriodicCollector::PeriodicCollector(Sampler* sampler,
MetricReportQueue* metric_report_queue,
ReportingSettings* reporting_settings,
const std::string& enable_setting_path,
bool setting_enabled_default_value,
const std::string& rate_setting_path,
base::TimeDelta default_rate,
int rate_unit_to_ms)
: PeriodicCollector(sampler,
metric_report_queue,
reporting_settings,
enable_setting_path,
setting_enabled_default_value,
rate_setting_path,
default_rate,
rate_unit_to_ms,
base::TimeDelta()) {}
PeriodicCollector::~PeriodicCollector() = default;
void PeriodicCollector::OnMetricDataCollected(
bool is_event_driven,
std::optional<MetricData> metric_data) {
CheckOnSequence();
CHECK(metric_report_queue_);
if (!metric_data.has_value()) {
base::UmaHistogramExactLinear(
PeriodicCollector::kNoMetricDataMetricsName,
static_cast<int>(metric_report_queue_->GetDestination()),
Destination_ARRAYSIZE);
return;
}
if (is_event_driven) {
CHECK(metric_data->has_telemetry_data());
metric_data->mutable_telemetry_data()->set_is_event_driven(is_event_driven);
}
metric_data->set_timestamp_ms(
base::Time::Now().InMillisecondsSinceUnixEpoch());
metric_report_queue_->Enqueue(std::move(metric_data.value()));
}
bool PeriodicCollector::CanCollect() const {
return reporting_controller_->IsEnabled();
}
void PeriodicCollector::StartPeriodicCollection() {
CheckOnSequence();
// Do initial collection at startup.
Collect(/*is_event_driven=*/false);
rate_controller_->Start();
}
void PeriodicCollector::StopPeriodicCollection() {
CheckOnSequence();
rate_controller_->Stop();
}
void PeriodicCollector::SetReportingControllerCb() {
reporting_controller_->SetSettingUpdateCb(
base::BindRepeating(&PeriodicCollector::StartPeriodicCollection,
base::Unretained(this)),
base::BindRepeating(&PeriodicCollector::StopPeriodicCollection,
base::Unretained(this)));
}
} // namespace reporting