| // 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 |