1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97
|
// Copyright 2023 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/ssl/daily_navigation_counter.h"
#include "base/i18n/time_formatting.h"
#include "base/time/clock.h"
#include "third_party/icu/source/i18n/unicode/timezone.h"
namespace {
// Returns the given time in yyyy-MM-dd format.
std::string GetDateString(base::Time now) {
return base::UnlocalizedTimeFormatWithPattern(now.UTCMidnight(), "yyyy-MM-dd",
icu::TimeZone::getGMT());
}
} // namespace
DailyNavigationCounter::DailyNavigationCounter(
base::Value::Dict* dict,
base::Clock* clock,
size_t rolling_window_duration_in_days,
size_t save_interval)
: dict_(dict),
clock_(clock),
rolling_window_duration_in_days_(rolling_window_duration_in_days),
save_interval_(save_interval) {
DCHECK(save_interval_ > 0);
// Load from the saved dict into the current map, discarding old entries.
base::Time now = clock_->Now();
base::Time cutoff =
now.UTCMidnight() - base::Days(rolling_window_duration_in_days_);
for (auto [key, value] : *dict_) {
base::Time timestamp;
if (!base::Time::FromUTCString(key.c_str(), ×tamp)) {
// Not a valid bucket.
continue;
}
if (timestamp < cutoff) {
// Old bucket.
continue;
}
std::optional<int> count = dict->FindInt(key);
if (!count.has_value()) {
continue;
}
counts_map_[key] = count.value();
}
}
DailyNavigationCounter::~DailyNavigationCounter() = default;
size_t DailyNavigationCounter::GetTotal() const {
size_t total = 0;
for (auto [key, value] : counts_map_) {
total += value;
}
return total;
}
bool DailyNavigationCounter::Increment() {
base::Time now = clock_->Now();
std::string today = GetDateString(now);
counts_map_[today]++;
unsaved_count_++;
if (unsaved_count_ < save_interval_) {
// Not stale.
return false;
}
// Otherwise, discard old buckets and save the current map to the backing
// dict.
base::Time cutoff =
now.UTCMidnight() - base::Days(rolling_window_duration_in_days_);
base::flat_map<std::string, int> new_map_;
dict_->clear();
for (auto [key, value] : counts_map_) {
base::Time timestamp;
if (!base::Time::FromUTCString(key.c_str(), ×tamp)) {
// Not a valid bucket.
continue;
}
if (timestamp < cutoff) {
// Old bucket.
continue;
}
new_map_[key] = value;
dict_->Set(key, value);
}
counts_map_ = new_map_;
unsaved_count_ = 0;
return true;
}
|