blob: 93fd1e686f4d33b2b99e350ca6440f6a2870d76a [file] [log] [blame]
// Copyright 2017 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 "net/reporting/reporting_cache.h"
#include <algorithm>
#include <string>
#include "base/strings/stringprintf.h"
#include "base/test/simple_test_tick_clock.h"
#include "base/test/values_test_util.h"
#include "base/time/time.h"
#include "base/values.h"
#include "net/reporting/reporting_client.h"
#include "net/reporting/reporting_observer.h"
#include "net/reporting/reporting_report.h"
#include "net/reporting/reporting_test_util.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "url/gurl.h"
#include "url/origin.h"
namespace net {
namespace {
class TestReportingObserver : public ReportingObserver {
public:
TestReportingObserver() : cache_update_count_(0) {}
void OnCacheUpdated() override { ++cache_update_count_; }
int cache_update_count() const { return cache_update_count_; }
private:
int cache_update_count_;
};
class ReportingCacheTest : public ReportingTestBase {
protected:
ReportingCacheTest() : ReportingTestBase() {
ReportingPolicy policy;
policy.max_report_count = 5;
policy.max_client_count = 5;
UsePolicy(policy);
context()->AddObserver(&observer_);
}
~ReportingCacheTest() override { context()->RemoveObserver(&observer_); }
TestReportingObserver* observer() { return &observer_; }
size_t report_count() {
std::vector<const ReportingReport*> reports;
cache()->GetReports(&reports);
return reports.size();
}
size_t client_count() {
std::vector<const ReportingClient*> clients;
cache()->GetClients(&clients);
return clients.size();
}
void SetClient(const url::Origin& origin,
const GURL& endpoint,
bool subdomains,
const std::string& group,
base::TimeTicks expires) {
cache()->SetClient(origin, endpoint,
subdomains ? ReportingClient::Subdomains::INCLUDE
: ReportingClient::Subdomains::EXCLUDE,
group, expires, ReportingClient::kDefaultPriority,
ReportingClient::kDefaultWeight);
}
// Adds a new report to the cache, and returns it.
const ReportingReport* AddAndReturnReport(
const GURL& url,
const std::string& user_agent,
const std::string& group,
const std::string& type,
std::unique_ptr<const base::Value> body,
int depth,
base::TimeTicks queued,
int attempts) {
const base::Value* body_unowned = body.get();
// The public API will only give us the (unordered) full list of reports in
// the cache. So we need to grab the list before we add, and the list after
// we add, and return the one element that's different. This is only used
// in test cases, so I've optimized for readability over execution speed.
std::vector<const ReportingReport*> before;
cache()->GetReports(&before);
cache()->AddReport(url, user_agent, group, type, std::move(body), depth,
queued, attempts);
std::vector<const ReportingReport*> after;
cache()->GetReports(&after);
for (const ReportingReport* report : after) {
// If report isn't in before, we've found the new instance.
if (std::find(before.begin(), before.end(), report) == before.end()) {
// Sanity check the result before we return it.
EXPECT_EQ(url, report->url);
EXPECT_EQ(user_agent, report->user_agent);
EXPECT_EQ(group, report->group);
EXPECT_EQ(type, report->type);
EXPECT_EQ(*body_unowned, *report->body);
EXPECT_EQ(depth, report->depth);
EXPECT_EQ(queued, report->queued);
EXPECT_EQ(attempts, report->attempts);
return report;
}
}
// This can actually happen! If the newly created report isn't in the after
// vector, that means that we had to evict a report, and the new report was
// the only one eligible for eviction!
return nullptr;
}
const GURL kUrl1_ = GURL("https://origin1/path");
const GURL kUrl2_ = GURL("https://origin2/path");
const url::Origin kOrigin1_ = url::Origin::Create(GURL("https://origin1/"));
const url::Origin kOrigin2_ = url::Origin::Create(GURL("https://origin2/"));
const GURL kEndpoint1_ = GURL("https://endpoint1/");
const GURL kEndpoint2_ = GURL("https://endpoint2/");
const std::string kUserAgent_ = "Mozilla/1.0";
const std::string kGroup1_ = "group1";
const std::string kGroup2 = "group2";
const std::string kType_ = "default";
const base::TimeTicks kNow_ = base::TimeTicks::Now();
const base::TimeTicks kExpires1_ = kNow_ + base::TimeDelta::FromDays(7);
const base::TimeTicks kExpires2_ = kExpires1_ + base::TimeDelta::FromDays(7);
private:
TestReportingObserver observer_;
};
TEST_F(ReportingCacheTest, Reports) {
std::vector<const ReportingReport*> reports;
cache()->GetReports(&reports);
EXPECT_TRUE(reports.empty());
cache()->AddReport(kUrl1_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 0, kNow_, 0);
EXPECT_EQ(1, observer()->cache_update_count());
cache()->GetReports(&reports);
ASSERT_EQ(1u, reports.size());
const ReportingReport* report = reports[0];
ASSERT_TRUE(report);
EXPECT_EQ(kUrl1_, report->url);
EXPECT_EQ(kUserAgent_, report->user_agent);
EXPECT_EQ(kGroup1_, report->group);
EXPECT_EQ(kType_, report->type);
// TODO(juliatuttle): Check body?
EXPECT_EQ(kNow_, report->queued);
EXPECT_EQ(0, report->attempts);
EXPECT_FALSE(cache()->IsReportPendingForTesting(report));
EXPECT_FALSE(cache()->IsReportDoomedForTesting(report));
cache()->IncrementReportsAttempts(reports);
EXPECT_EQ(2, observer()->cache_update_count());
cache()->GetReports(&reports);
ASSERT_EQ(1u, reports.size());
report = reports[0];
ASSERT_TRUE(report);
EXPECT_EQ(1, report->attempts);
cache()->RemoveReports(reports, ReportingReport::Outcome::UNKNOWN);
EXPECT_EQ(3, observer()->cache_update_count());
cache()->GetReports(&reports);
EXPECT_TRUE(reports.empty());
}
TEST_F(ReportingCacheTest, RemoveAllReports) {
cache()->AddReport(kUrl1_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 0, kNow_, 0);
cache()->AddReport(kUrl1_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 0, kNow_, 0);
EXPECT_EQ(2, observer()->cache_update_count());
std::vector<const ReportingReport*> reports;
cache()->GetReports(&reports);
EXPECT_EQ(2u, reports.size());
cache()->RemoveAllReports(ReportingReport::Outcome::UNKNOWN);
EXPECT_EQ(3, observer()->cache_update_count());
cache()->GetReports(&reports);
EXPECT_TRUE(reports.empty());
}
TEST_F(ReportingCacheTest, RemovePendingReports) {
cache()->AddReport(kUrl1_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 0, kNow_, 0);
EXPECT_EQ(1, observer()->cache_update_count());
std::vector<const ReportingReport*> reports;
cache()->GetReports(&reports);
ASSERT_EQ(1u, reports.size());
EXPECT_FALSE(cache()->IsReportPendingForTesting(reports[0]));
EXPECT_FALSE(cache()->IsReportDoomedForTesting(reports[0]));
cache()->SetReportsPending(reports);
EXPECT_TRUE(cache()->IsReportPendingForTesting(reports[0]));
EXPECT_FALSE(cache()->IsReportDoomedForTesting(reports[0]));
cache()->RemoveReports(reports, ReportingReport::Outcome::UNKNOWN);
EXPECT_TRUE(cache()->IsReportPendingForTesting(reports[0]));
EXPECT_TRUE(cache()->IsReportDoomedForTesting(reports[0]));
EXPECT_EQ(2, observer()->cache_update_count());
// After removing report, future calls to GetReports should not return it.
std::vector<const ReportingReport*> visible_reports;
cache()->GetReports(&visible_reports);
EXPECT_TRUE(visible_reports.empty());
EXPECT_EQ(1u, cache()->GetFullReportCountForTesting());
// After clearing pending flag, report should be deleted.
cache()->ClearReportsPending(reports);
EXPECT_EQ(0u, cache()->GetFullReportCountForTesting());
}
TEST_F(ReportingCacheTest, RemoveAllPendingReports) {
cache()->AddReport(kUrl1_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 0, kNow_, 0);
EXPECT_EQ(1, observer()->cache_update_count());
std::vector<const ReportingReport*> reports;
cache()->GetReports(&reports);
ASSERT_EQ(1u, reports.size());
EXPECT_FALSE(cache()->IsReportPendingForTesting(reports[0]));
EXPECT_FALSE(cache()->IsReportDoomedForTesting(reports[0]));
cache()->SetReportsPending(reports);
EXPECT_TRUE(cache()->IsReportPendingForTesting(reports[0]));
EXPECT_FALSE(cache()->IsReportDoomedForTesting(reports[0]));
cache()->RemoveAllReports(ReportingReport::Outcome::UNKNOWN);
EXPECT_TRUE(cache()->IsReportPendingForTesting(reports[0]));
EXPECT_TRUE(cache()->IsReportDoomedForTesting(reports[0]));
EXPECT_EQ(2, observer()->cache_update_count());
// After removing report, future calls to GetReports should not return it.
std::vector<const ReportingReport*> visible_reports;
cache()->GetReports(&visible_reports);
EXPECT_TRUE(visible_reports.empty());
EXPECT_EQ(1u, cache()->GetFullReportCountForTesting());
// After clearing pending flag, report should be deleted.
cache()->ClearReportsPending(reports);
EXPECT_EQ(0u, cache()->GetFullReportCountForTesting());
}
TEST_F(ReportingCacheTest, GetReportsAsValue) {
// We need a reproducible expiry timestamp for this test case.
const base::TimeTicks now = base::TimeTicks();
const ReportingReport* report1 =
AddAndReturnReport(kUrl1_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 0,
now + base::TimeDelta::FromSeconds(200), 0);
const ReportingReport* report2 =
AddAndReturnReport(kUrl1_, kUserAgent_, kGroup2, kType_,
std::make_unique<base::DictionaryValue>(), 0,
now + base::TimeDelta::FromSeconds(100), 1);
cache()->AddReport(kUrl2_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 2,
now + base::TimeDelta::FromSeconds(200), 0);
cache()->AddReport(kUrl1_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 0,
now + base::TimeDelta::FromSeconds(300), 0);
// Mark report1 as pending as report2 as doomed
cache()->SetReportsPending({report1, report2});
cache()->RemoveReports({report2}, ReportingReport::Outcome::UNKNOWN);
base::Value actual = cache()->GetReportsAsValue();
std::unique_ptr<base::Value> expected = base::test::ParseJson(R"json(
[
{
"url": "https://origin1/path",
"group": "group2",
"type": "default",
"status": "doomed",
"body": {},
"attempts": 1,
"depth": 0,
"queued": "100000",
},
{
"url": "https://origin1/path",
"group": "group1",
"type": "default",
"status": "pending",
"body": {},
"attempts": 0,
"depth": 0,
"queued": "200000",
},
{
"url": "https://origin2/path",
"group": "group1",
"type": "default",
"status": "queued",
"body": {},
"attempts": 0,
"depth": 2,
"queued": "200000",
},
{
"url": "https://origin1/path",
"group": "group1",
"type": "default",
"status": "queued",
"body": {},
"attempts": 0,
"depth": 0,
"queued": "300000",
},
]
)json");
EXPECT_EQ(*expected, actual);
}
TEST_F(ReportingCacheTest, Endpoints) {
SetClient(kOrigin1_, kEndpoint1_, false, kGroup1_, kExpires1_);
EXPECT_EQ(1, observer()->cache_update_count());
const ReportingClient* client =
FindClientInCache(cache(), kOrigin1_, kEndpoint1_);
ASSERT_TRUE(client);
EXPECT_EQ(kOrigin1_, client->origin);
EXPECT_EQ(kEndpoint1_, client->endpoint);
EXPECT_EQ(ReportingClient::Subdomains::EXCLUDE, client->subdomains);
EXPECT_EQ(kGroup1_, client->group);
EXPECT_EQ(kExpires1_, client->expires);
SetClient(kOrigin1_, kEndpoint1_, true, kGroup2, kExpires2_);
EXPECT_EQ(2, observer()->cache_update_count());
client = FindClientInCache(cache(), kOrigin1_, kEndpoint1_);
ASSERT_TRUE(client);
EXPECT_EQ(kOrigin1_, client->origin);
EXPECT_EQ(kEndpoint1_, client->endpoint);
EXPECT_EQ(ReportingClient::Subdomains::INCLUDE, client->subdomains);
EXPECT_EQ(kGroup2, client->group);
EXPECT_EQ(kExpires2_, client->expires);
cache()->RemoveClients(std::vector<const ReportingClient*>{client});
EXPECT_EQ(3, observer()->cache_update_count());
client = FindClientInCache(cache(), kOrigin1_, kEndpoint1_);
EXPECT_FALSE(client);
}
TEST_F(ReportingCacheTest, GetClientsForOriginAndGroup) {
SetClient(kOrigin1_, kEndpoint1_, false, kGroup1_, kExpires1_);
SetClient(kOrigin1_, kEndpoint2_, false, kGroup2, kExpires1_);
SetClient(kOrigin2_, kEndpoint1_, false, kGroup1_, kExpires1_);
std::vector<const ReportingClient*> clients;
cache()->GetClientsForOriginAndGroup(kOrigin1_, kGroup1_, &clients);
ASSERT_EQ(1u, clients.size());
const ReportingClient* client = clients[0];
ASSERT_TRUE(client);
EXPECT_EQ(kOrigin1_, client->origin);
EXPECT_EQ(kGroup1_, client->group);
}
TEST_F(ReportingCacheTest, RemoveClientForOriginAndEndpoint) {
SetClient(kOrigin1_, kEndpoint1_, false, kGroup1_, kExpires1_);
SetClient(kOrigin1_, kEndpoint2_, false, kGroup2, kExpires1_);
SetClient(kOrigin2_, kEndpoint1_, false, kGroup1_, kExpires1_);
EXPECT_EQ(3, observer()->cache_update_count());
cache()->RemoveClientForOriginAndEndpoint(kOrigin1_, kEndpoint1_);
EXPECT_EQ(4, observer()->cache_update_count());
std::vector<const ReportingClient*> clients;
cache()->GetClientsForOriginAndGroup(kOrigin1_, kGroup1_, &clients);
EXPECT_TRUE(clients.empty());
cache()->GetClientsForOriginAndGroup(kOrigin1_, kGroup2, &clients);
EXPECT_EQ(1u, clients.size());
cache()->GetClientsForOriginAndGroup(kOrigin2_, kGroup1_, &clients);
EXPECT_EQ(1u, clients.size());
}
TEST_F(ReportingCacheTest, RemoveClientsForEndpoint) {
SetClient(kOrigin1_, kEndpoint1_, false, kGroup1_, kExpires1_);
SetClient(kOrigin1_, kEndpoint2_, false, kGroup2, kExpires1_);
SetClient(kOrigin2_, kEndpoint1_, false, kGroup1_, kExpires1_);
EXPECT_EQ(3, observer()->cache_update_count());
cache()->RemoveClientsForEndpoint(kEndpoint1_);
EXPECT_EQ(4, observer()->cache_update_count());
std::vector<const ReportingClient*> clients;
cache()->GetClientsForOriginAndGroup(kOrigin1_, kGroup1_, &clients);
EXPECT_TRUE(clients.empty());
cache()->GetClientsForOriginAndGroup(kOrigin1_, kGroup2, &clients);
EXPECT_EQ(1u, clients.size());
cache()->GetClientsForOriginAndGroup(kOrigin2_, kGroup1_, &clients);
EXPECT_TRUE(clients.empty());
}
TEST_F(ReportingCacheTest, GetClientsAsValue) {
// We need a reproducible expiry timestamp for this test case.
const base::TimeTicks expires =
base::TimeTicks() + base::TimeDelta::FromDays(7);
SetClient(kOrigin1_, kEndpoint1_, false, kGroup1_, expires);
SetClient(kOrigin2_, kEndpoint2_, true, kGroup1_, expires);
cache()->IncrementEndpointDeliveries(kOrigin1_, kEndpoint1_, 2, true);
cache()->IncrementEndpointDeliveries(kOrigin2_, kEndpoint2_, 1, false);
base::Value actual = cache()->GetClientsAsValue();
std::unique_ptr<base::Value> expected = base::test::ParseJson(R"json(
[
{
"origin": "https://origin1",
"groups": [
{
"name": "group1",
"expires": "604800000",
"includeSubdomains": false,
"endpoints": [
{"url": "https://endpoint1/", "priority": 0, "weight": 1,
"successful": {"uploads": 1, "reports": 2},
"failed": {"uploads": 0, "reports": 0}},
],
},
],
},
{
"origin": "https://origin2",
"groups": [
{
"name": "group1",
"expires": "604800000",
"includeSubdomains": true,
"endpoints": [
{"url": "https://endpoint2/", "priority": 0, "weight": 1,
"successful": {"uploads": 0, "reports": 0},
"failed": {"uploads": 1, "reports": 1}},
],
},
],
},
]
)json");
EXPECT_EQ(*expected, actual);
}
TEST_F(ReportingCacheTest, RemoveAllClients) {
SetClient(kOrigin1_, kEndpoint1_, false, kGroup1_, kExpires1_);
SetClient(kOrigin2_, kEndpoint2_, false, kGroup1_, kExpires1_);
EXPECT_EQ(2, observer()->cache_update_count());
cache()->RemoveAllClients();
EXPECT_EQ(3, observer()->cache_update_count());
std::vector<const ReportingClient*> clients;
cache()->GetClients(&clients);
EXPECT_TRUE(clients.empty());
}
TEST_F(ReportingCacheTest, ExcludeSubdomainsDifferentPort) {
const url::Origin kOrigin = url::Origin::Create(GURL("https://example/"));
const url::Origin kDifferentPortOrigin =
url::Origin::Create(GURL("https://example:444/"));
SetClient(kDifferentPortOrigin, kEndpoint1_, false, kGroup1_, kExpires1_);
std::vector<const ReportingClient*> clients;
cache()->GetClientsForOriginAndGroup(kOrigin, kGroup1_, &clients);
EXPECT_TRUE(clients.empty());
}
TEST_F(ReportingCacheTest, ExcludeSubdomainsSuperdomain) {
const url::Origin kOrigin = url::Origin::Create(GURL("https://foo.example/"));
const url::Origin kSuperOrigin =
url::Origin::Create(GURL("https://example/"));
SetClient(kSuperOrigin, kEndpoint1_, false, kGroup1_, kExpires1_);
std::vector<const ReportingClient*> clients;
cache()->GetClientsForOriginAndGroup(kOrigin, kGroup1_, &clients);
EXPECT_TRUE(clients.empty());
}
TEST_F(ReportingCacheTest, IncludeSubdomainsDifferentPort) {
const url::Origin kOrigin = url::Origin::Create(GURL("https://example/"));
const url::Origin kDifferentPortOrigin =
url::Origin::Create(GURL("https://example:444/"));
SetClient(kDifferentPortOrigin, kEndpoint1_, true, kGroup1_, kExpires1_);
std::vector<const ReportingClient*> clients;
cache()->GetClientsForOriginAndGroup(kOrigin, kGroup1_, &clients);
ASSERT_EQ(1u, clients.size());
EXPECT_EQ(kDifferentPortOrigin, clients[0]->origin);
}
TEST_F(ReportingCacheTest, IncludeSubdomainsSuperdomain) {
const url::Origin kOrigin = url::Origin::Create(GURL("https://foo.example/"));
const url::Origin kSuperOrigin =
url::Origin::Create(GURL("https://example/"));
SetClient(kSuperOrigin, kEndpoint1_, true, kGroup1_, kExpires1_);
std::vector<const ReportingClient*> clients;
cache()->GetClientsForOriginAndGroup(kOrigin, kGroup1_, &clients);
ASSERT_EQ(1u, clients.size());
EXPECT_EQ(kSuperOrigin, clients[0]->origin);
}
TEST_F(ReportingCacheTest, IncludeSubdomainsPreferOriginToDifferentPort) {
const url::Origin kOrigin = url::Origin::Create(GURL("https://foo.example/"));
const url::Origin kDifferentPortOrigin =
url::Origin::Create(GURL("https://example:444/"));
SetClient(kOrigin, kEndpoint1_, true, kGroup1_, kExpires1_);
SetClient(kDifferentPortOrigin, kEndpoint1_, true, kGroup1_, kExpires1_);
std::vector<const ReportingClient*> clients;
cache()->GetClientsForOriginAndGroup(kOrigin, kGroup1_, &clients);
ASSERT_EQ(1u, clients.size());
EXPECT_EQ(kOrigin, clients[0]->origin);
}
TEST_F(ReportingCacheTest, IncludeSubdomainsPreferOriginToSuperdomain) {
const url::Origin kOrigin = url::Origin::Create(GURL("https://foo.example/"));
const url::Origin kSuperOrigin =
url::Origin::Create(GURL("https://example/"));
SetClient(kOrigin, kEndpoint1_, true, kGroup1_, kExpires1_);
SetClient(kSuperOrigin, kEndpoint1_, true, kGroup1_, kExpires1_);
std::vector<const ReportingClient*> clients;
cache()->GetClientsForOriginAndGroup(kOrigin, kGroup1_, &clients);
ASSERT_EQ(1u, clients.size());
EXPECT_EQ(kOrigin, clients[0]->origin);
}
TEST_F(ReportingCacheTest, IncludeSubdomainsPreferMoreSpecificSuperdomain) {
const url::Origin kOrigin =
url::Origin::Create(GURL("https://foo.bar.example/"));
const url::Origin kSuperOrigin =
url::Origin::Create(GURL("https://bar.example/"));
const url::Origin kSuperSuperOrigin =
url::Origin::Create(GURL("https://example/"));
SetClient(kSuperOrigin, kEndpoint1_, true, kGroup1_, kExpires1_);
SetClient(kSuperSuperOrigin, kEndpoint1_, true, kGroup1_, kExpires1_);
std::vector<const ReportingClient*> clients;
cache()->GetClientsForOriginAndGroup(kOrigin, kGroup1_, &clients);
ASSERT_EQ(1u, clients.size());
EXPECT_EQ(kSuperOrigin, clients[0]->origin);
}
TEST_F(ReportingCacheTest, EvictOldestReport) {
size_t max_report_count = policy().max_report_count;
ASSERT_LT(0u, max_report_count);
ASSERT_GT(std::numeric_limits<size_t>::max(), max_report_count);
base::TimeTicks earliest_queued = tick_clock()->NowTicks();
// Enqueue the maximum number of reports, spaced apart in time.
for (size_t i = 0; i < max_report_count; ++i) {
cache()->AddReport(kUrl1_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 0,
tick_clock()->NowTicks(), 0);
tick_clock()->Advance(base::TimeDelta::FromMinutes(1));
}
EXPECT_EQ(max_report_count, report_count());
// Add one more report to force the cache to evict one.
cache()->AddReport(kUrl1_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 0, kNow_, 0);
// Make sure the cache evicted a report to make room for the new one, and make
// sure the report evicted was the earliest-queued one.
std::vector<const ReportingReport*> reports;
cache()->GetReports(&reports);
EXPECT_EQ(max_report_count, reports.size());
for (const ReportingReport* report : reports)
EXPECT_NE(earliest_queued, report->queued);
}
TEST_F(ReportingCacheTest, DontEvictPendingReports) {
size_t max_report_count = policy().max_report_count;
ASSERT_LT(0u, max_report_count);
ASSERT_GT(std::numeric_limits<size_t>::max(), max_report_count);
// Enqueue the maximum number of reports, spaced apart in time.
for (size_t i = 0; i < max_report_count; ++i) {
cache()->AddReport(kUrl1_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 0,
tick_clock()->NowTicks(), 0);
tick_clock()->Advance(base::TimeDelta::FromMinutes(1));
}
EXPECT_EQ(max_report_count, report_count());
// Mark all of the queued reports pending.
std::vector<const ReportingReport*> queued_reports;
cache()->GetReports(&queued_reports);
cache()->SetReportsPending(queued_reports);
// Add one more report to force the cache to evict one. Since the cache has
// only pending reports, it will be forced to evict the *new* report!
cache()->AddReport(kUrl1_, kUserAgent_, kGroup1_, kType_,
std::make_unique<base::DictionaryValue>(), 0, kNow_, 0);
// Make sure the cache evicted a report, and make sure the report evicted was
// the new, non-pending one.
std::vector<const ReportingReport*> reports;
cache()->GetReports(&reports);
EXPECT_EQ(max_report_count, reports.size());
for (const ReportingReport* report : reports)
EXPECT_TRUE(cache()->IsReportPendingForTesting(report));
}
GURL MakeEndpoint(size_t index) {
return GURL(base::StringPrintf("https://endpoint/%zd", index));
}
TEST_F(ReportingCacheTest, EvictLRUClient) {
size_t max_client_count = policy().max_client_count;
ASSERT_LT(0u, max_client_count);
ASSERT_GT(std::numeric_limits<size_t>::max(), max_client_count);
for (size_t i = 0; i < max_client_count; ++i) {
SetClient(kOrigin1_, MakeEndpoint(i), false, kGroup1_, tomorrow());
}
EXPECT_EQ(max_client_count, client_count());
// Use clients in reverse order, so client (max_client_count - 1) is LRU.
for (size_t i = 1; i <= max_client_count; ++i) {
const ReportingClient* client = FindClientInCache(
cache(), kOrigin1_, MakeEndpoint(max_client_count - i));
cache()->MarkClientUsed(client);
tick_clock()->Advance(base::TimeDelta::FromSeconds(1));
}
// Add one more client, forcing the cache to evict the LRU.
SetClient(kOrigin1_, MakeEndpoint(max_client_count), false, kGroup1_,
tomorrow());
EXPECT_EQ(max_client_count, client_count());
EXPECT_FALSE(FindClientInCache(cache(), kOrigin1_,
MakeEndpoint(max_client_count - 1)));
}
TEST_F(ReportingCacheTest, EvictExpiredClient) {
size_t max_client_count = policy().max_client_count;
ASSERT_LT(0u, max_client_count);
ASSERT_GT(std::numeric_limits<size_t>::max(), max_client_count);
for (size_t i = 0; i < max_client_count; ++i) {
base::TimeTicks expires =
(i == max_client_count - 1) ? yesterday() : tomorrow();
SetClient(kOrigin1_, MakeEndpoint(i), false, kGroup1_, expires);
}
EXPECT_EQ(max_client_count, client_count());
// Add one more client, forcing the cache to evict the expired one.
SetClient(kOrigin1_, MakeEndpoint(max_client_count), false, kGroup1_,
tomorrow());
EXPECT_EQ(max_client_count, client_count());
EXPECT_FALSE(FindClientInCache(cache(), kOrigin1_,
MakeEndpoint(max_client_count - 1)));
}
} // namespace
} // namespace net