blob: d436bfe4bb923ee35418dcca986c9ca82b65b70c [file] [log] [blame]
// Copyright 2020 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 "chrome/browser/web_applications/daily_metrics_helper.h"
#include "base/numerics/ranges.h"
#include "chrome/browser/metrics/ukm_background_recorder_service.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/common/pref_names.h"
#include "components/metrics/date_changed_helper.h"
#include "components/prefs/pref_registry_simple.h"
#include "components/prefs/pref_service.h"
#include "components/prefs/scoped_user_pref_update.h"
#include "content/public/browser/browser_thread.h"
#include "services/metrics/public/cpp/metrics_utils.h"
#include "services/metrics/public/cpp/ukm_builders.h"
#include "url/origin.h"
namespace web_app {
namespace {
int BucketedDailySeconds(base::TimeDelta delta) {
int64_t sample = base::ClampToRange(delta.InSeconds(), int64_t(0),
base::TimeDelta::FromDays(1).InSeconds());
// Result between 1 sec and 1 day, in 50 linear buckets per day.
int32_t bucket_size = base::TimeDelta::FromDays(1).InSeconds() / 50;
int result = ukm::GetLinearBucketMin(sample, bucket_size);
return std::max(1, result);
}
} // namespace
// This class exists just to be friended by |UkmRecorder|.
class DesktopWebAppUkmRecorder {
public:
static void Emit(DailyInteraction record) {
DCHECK(record.start_url.is_valid());
ukm::SourceId source_id =
ukm::UkmRecorder::GetSourceIdForDesktopWebAppStartUrl(record.start_url);
ukm::builders::WebApp_DailyInteraction builder(source_id);
builder.SetUsed(true)
.SetInstalled(record.installed)
.SetDisplayMode(record.effective_display_mode)
.SetPromotable(record.promotable);
if (record.install_source)
builder.SetInstallSource(record.install_source.value());
if (!record.foreground_duration.is_zero())
builder.SetForegroundDuration(
BucketedDailySeconds(record.foreground_duration));
if (!record.background_duration.is_zero())
builder.SetBackgroundDuration(
BucketedDailySeconds(record.background_duration));
if (record.num_sessions > 0)
builder.SetNumSessions(record.num_sessions);
builder.Record(ukm::UkmRecorder::Get());
}
};
namespace {
using base::DictionaryValue;
using base::Optional;
using base::TimeDelta;
using base::Value;
bool skip_origin_check_for_testing_ = false;
const char kInstalled[] = "installed";
const char kInstallSource[] = "install_source";
const char kEffectiveDisplayMode[] = "effective_display_mode";
const char kPromotable[] = "promotable";
const char kForegroundDurationSec[] = "foreground_duration_sec";
const char kBackgroundDurationSec[] = "background_duration_sec";
const char kNumSessions[] = "num_sessions";
Optional<DailyInteraction> DictToRecord(const std::string& url,
const DictionaryValue& record_dict) {
GURL gurl(url);
if (!gurl.is_valid())
return base::nullopt;
DailyInteraction record(gurl);
Optional<int> installed = record_dict.FindBoolKey(kInstalled);
if (!installed.has_value())
return base::nullopt;
record.installed = *installed;
record.install_source = record_dict.FindIntKey(kInstallSource);
Optional<int> effective_display_mode =
record_dict.FindIntKey(kEffectiveDisplayMode);
if (!effective_display_mode.has_value())
return base::nullopt;
record.effective_display_mode = *effective_display_mode;
Optional<bool> promotable = record_dict.FindBoolKey(kPromotable);
if (!promotable.has_value())
return base::nullopt;
record.promotable = *promotable;
Optional<int> foreground_duration_sec =
record_dict.FindIntKey(kForegroundDurationSec);
if (foreground_duration_sec) {
record.foreground_duration =
TimeDelta::FromSeconds(*foreground_duration_sec);
}
Optional<int> background_duration_sec =
record_dict.FindIntKey(kBackgroundDurationSec);
if (background_duration_sec) {
record.background_duration =
TimeDelta::FromSeconds(*background_duration_sec);
}
Optional<int> num_sessions = record_dict.FindIntKey(kNumSessions);
if (num_sessions)
record.num_sessions = *num_sessions;
return record;
}
std::unique_ptr<DictionaryValue> RecordToDict(DailyInteraction& record) {
auto record_dict = std::make_unique<DictionaryValue>();
// Note URL is not set here as it is the key for this dict in its parent.
record_dict->SetBoolKey(kInstalled, record.installed);
if (record.install_source.has_value())
record_dict->SetIntKey(kInstallSource, *record.install_source);
record_dict->SetIntKey(kEffectiveDisplayMode, record.effective_display_mode);
record_dict->SetBoolKey(kPromotable, record.promotable);
record_dict->SetIntKey(kForegroundDurationSec,
record.foreground_duration.InSeconds());
record_dict->SetIntKey(kBackgroundDurationSec,
record.background_duration.InSeconds());
record_dict->SetIntKey(kNumSessions, record.num_sessions);
return record_dict;
}
void EmitIfSourceIdExists(DailyInteraction record,
Optional<ukm::SourceId> origin_source_id) {
if (!origin_source_id)
return;
// Make our own source ID that captures entire start_url, not just origin.
DesktopWebAppUkmRecorder::Emit(record);
}
void EmitRecord(DailyInteraction record, Profile* profile) {
if (skip_origin_check_for_testing_) {
DesktopWebAppUkmRecorder::Emit(record);
return;
}
auto* ukm_background_service =
ukm::UkmBackgroundRecorderFactory::GetForProfile(profile);
url::Origin origin = url::Origin::Create(record.start_url);
// Ensure origin is still in the history before emitting.
ukm_background_service->GetBackgroundSourceIdIfAllowed(
origin, base::BindOnce(&EmitIfSourceIdExists, record));
}
void EmitRecords(Profile* profile) {
const DictionaryValue* urls_to_features =
profile->GetPrefs()->GetDictionary(prefs::kWebAppsDailyMetrics);
DCHECK(urls_to_features);
for (DictionaryValue::Iterator iter(*urls_to_features); !iter.IsAtEnd();
iter.Advance()) {
const std::string& url = iter.key();
const Value& val = iter.value();
const DictionaryValue& dict = Value::AsDictionaryValue(val);
Optional<DailyInteraction> record = DictToRecord(url, dict);
if (record)
EmitRecord(*record, profile);
}
}
void RemoveRecords(PrefService* prefs) {
const DictionaryValue* urls_to_features =
prefs->GetDictionary(prefs::kWebAppsDailyMetrics);
if (!urls_to_features)
return;
DictionaryPrefUpdate update(prefs, prefs::kWebAppsDailyMetrics);
update->Clear();
}
void UpdateRecord(DailyInteraction& record, PrefService* prefs) {
DCHECK(record.start_url.is_valid());
const std::string& url = record.start_url.spec();
const DictionaryValue* urls_to_features =
prefs->GetDictionary(prefs::kWebAppsDailyMetrics);
CHECK(urls_to_features);
const Value* existing_val = urls_to_features->FindDictKey(url);
if (existing_val) {
// Sum duration and session values from existing record.
const DictionaryValue& dict = Value::AsDictionaryValue(*existing_val);
Optional<DailyInteraction> existing_record = DictToRecord(url, dict);
if (existing_record) {
record.foreground_duration += existing_record->foreground_duration;
record.background_duration += existing_record->background_duration;
record.num_sessions += existing_record->num_sessions;
}
}
std::unique_ptr<DictionaryValue> record_dict = RecordToDict(record);
DictionaryPrefUpdate update(prefs, prefs::kWebAppsDailyMetrics);
update->SetKey(url, std::move(*record_dict));
}
} // namespace
DailyInteraction::DailyInteraction() = default;
DailyInteraction::DailyInteraction(GURL start_url) : start_url(start_url) {}
DailyInteraction::DailyInteraction(const DailyInteraction&) = default;
DailyInteraction::~DailyInteraction() = default;
void FlushOldRecordsAndUpdate(DailyInteraction& record, Profile* profile) {
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
if (metrics::date_changed_helper::HasDateChangedSinceLastCall(
profile->GetPrefs(), prefs::kWebAppsDailyMetricsDate)) {
EmitRecords(profile);
RemoveRecords(profile->GetPrefs());
}
UpdateRecord(record, profile->GetPrefs());
}
void FlushAllRecordsForTesting(Profile* profile) {
DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
EmitRecords(profile);
RemoveRecords(profile->GetPrefs());
}
void SkipOriginCheckForTesting() {
skip_origin_check_for_testing_ = true;
}
void RegisterDailyWebAppMetricsProfilePrefs(PrefRegistrySimple* registry) {
registry->RegisterDictionaryPref(prefs::kWebAppsDailyMetrics);
metrics::date_changed_helper::RegisterPref(registry,
prefs::kWebAppsDailyMetricsDate);
}
} // namespace web_app