blob: 3dc13c9b8aa26a63cadc372ebd1c32824386f27f [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 "content/browser/conversions/conversion_test_utils.h"
#include <limits.h>
#include <tuple>
#include "base/bind.h"
#include "base/bind_helpers.h"
#include "base/callback.h"
#include "base/run_loop.h"
#include "base/task_runner_util.h"
#include "base/test/bind_test_util.h"
#include "url/gurl.h"
namespace content {
namespace {
const char kDefaultImpressionOrigin[] = "https://impression.test/";
const char kDefaultConversionOrigin[] = "https://conversion.test/";
const char kDefaultReportOrigin[] = "https://report.test/";
// Default expiry time for impressions for testing.
const int64_t kExpiryTime = 30;
} // namespace
bool ConversionDisallowingContentBrowserClient::AllowConversionMeasurement(
BrowserContext* context) {
return false;
}
ConfigurableStorageDelegate::ConfigurableStorageDelegate() = default;
ConfigurableStorageDelegate::~ConfigurableStorageDelegate() = default;
void ConfigurableStorageDelegate::ProcessNewConversionReports(
std::vector<ConversionReport>* reports) {
// Note: reports are ordered by impression time, descending.
for (auto& report : *reports) {
report.report_time = report.impression.impression_time() +
base::TimeDelta::FromMilliseconds(report_time_ms_);
// If attribution credits were provided, associate them with reports
// in order.
if (!attribution_credits_.empty()) {
report.attribution_credit = attribution_credits_.front();
attribution_credits_.pop_front();
}
}
}
int ConfigurableStorageDelegate::GetMaxConversionsPerImpression() const {
return max_conversions_per_impression_;
}
int ConfigurableStorageDelegate::GetMaxImpressionsPerOrigin() const {
return max_impressions_per_origin_;
}
int ConfigurableStorageDelegate::GetMaxConversionsPerOrigin() const {
return max_conversions_per_origin_;
}
ConversionManager* TestManagerProvider::GetManager(
WebContents* web_contents) const {
return manager_;
}
TestConversionManager::TestConversionManager() = default;
TestConversionManager::~TestConversionManager() = default;
void TestConversionManager::HandleImpression(
const StorableImpression& impression) {
num_impressions_++;
}
void TestConversionManager::HandleConversion(
const StorableConversion& conversion) {
num_conversions_++;
}
void TestConversionManager::GetActiveImpressionsForWebUI(
base::OnceCallback<void(std::vector<StorableImpression>)> callback) {
std::move(callback).Run(impressions_);
}
void TestConversionManager::GetReportsForWebUI(
base::OnceCallback<void(std::vector<ConversionReport>)> callback,
base::Time max_report_time) {
std::move(callback).Run(reports_);
}
void TestConversionManager::SendReportsForWebUI(base::OnceClosure done) {
reports_.clear();
std::move(done).Run();
}
const ConversionPolicy& TestConversionManager::GetConversionPolicy() const {
return policy_;
}
void TestConversionManager::ClearData(
base::Time delete_begin,
base::Time delete_end,
base::RepeatingCallback<bool(const url::Origin&)> filter,
base::OnceClosure done) {
impressions_.clear();
reports_.clear();
std::move(done).Run();
}
void TestConversionManager::SetActiveImpressionsForWebUI(
std::vector<StorableImpression> impressions) {
impressions_ = std::move(impressions);
}
void TestConversionManager::SetReportsForWebUI(
std::vector<ConversionReport> reports) {
reports_ = std::move(reports);
}
void TestConversionManager::Reset() {
num_impressions_ = 0u;
num_conversions_ = 0u;
}
// Builds an impression with default values. This is done as a builder because
// all values needed to be provided at construction time.
ImpressionBuilder::ImpressionBuilder(base::Time time)
: impression_data_("123"),
impression_time_(time),
expiry_(base::TimeDelta::FromMilliseconds(kExpiryTime)),
impression_origin_(url::Origin::Create(GURL(kDefaultImpressionOrigin))),
conversion_origin_(url::Origin::Create(GURL(kDefaultConversionOrigin))),
reporting_origin_(url::Origin::Create(GURL(kDefaultReportOrigin))) {}
ImpressionBuilder::~ImpressionBuilder() = default;
ImpressionBuilder& ImpressionBuilder::SetExpiry(base::TimeDelta delta) {
expiry_ = delta;
return *this;
}
ImpressionBuilder& ImpressionBuilder::SetData(const std::string& data) {
impression_data_ = data;
return *this;
}
ImpressionBuilder& ImpressionBuilder::SetImpressionOrigin(
const url::Origin& origin) {
impression_origin_ = origin;
return *this;
}
ImpressionBuilder& ImpressionBuilder::SetConversionOrigin(
const url::Origin& origin) {
conversion_origin_ = origin;
return *this;
}
ImpressionBuilder& ImpressionBuilder::SetReportingOrigin(
const url::Origin& origin) {
reporting_origin_ = origin;
return *this;
}
StorableImpression ImpressionBuilder::Build() const {
return StorableImpression(impression_data_, impression_origin_,
conversion_origin_, reporting_origin_,
impression_time_,
impression_time_ + expiry_ /* expiry_time */,
base::nullopt /* impression_id */);
}
StorableConversion DefaultConversion() {
StorableConversion conversion(
"111" /* conversion_data */,
url::Origin::Create(
GURL(kDefaultConversionOrigin)) /* conversion_origin */,
url::Origin::Create(GURL(kDefaultReportOrigin)) /* reporting_origin */);
return conversion;
}
// Custom comparator for StorableImpressions that does not take impression id's
// into account.
testing::AssertionResult ImpressionsEqual(const StorableImpression& expected,
const StorableImpression& actual) {
const auto tie = [](const StorableImpression& impression) {
return std::make_tuple(
impression.impression_data(), impression.impression_origin(),
impression.conversion_origin(), impression.reporting_origin(),
impression.impression_time(), impression.expiry_time());
};
if (tie(expected) != tie(actual)) {
return testing::AssertionFailure();
}
return testing::AssertionSuccess();
}
// Custom comparator for comparing two vectors of conversion reports. Does not
// compare impression and conversion id's as they are set by the underlying
// sqlite db and should not be tested.
testing::AssertionResult ReportsEqual(
const std::vector<ConversionReport>& expected,
const std::vector<ConversionReport>& actual) {
const auto tie = [](const ConversionReport& conversion) {
return std::make_tuple(conversion.impression.impression_data(),
conversion.impression.impression_origin(),
conversion.impression.conversion_origin(),
conversion.impression.reporting_origin(),
conversion.impression.impression_time(),
conversion.impression.expiry_time(),
conversion.conversion_data, conversion.report_time,
conversion.attribution_credit);
};
if (expected.size() != actual.size())
return testing::AssertionFailure() << "Expected length " << expected.size()
<< ", actual: " << actual.size();
for (size_t i = 0; i < expected.size(); i++) {
if (tie(expected[i]) != tie(actual[i])) {
return testing::AssertionFailure()
<< "Expected " << expected[i] << " at index " << i
<< ", actual: " << actual[i];
}
}
return testing::AssertionSuccess();
}
std::vector<ConversionReport> GetConversionsToReportForTesting(
ConversionManagerImpl* manager,
base::Time max_report_time) {
base::RunLoop run_loop;
std::vector<ConversionReport> conversion_reports;
base::PostTaskAndReplyWithResult(
manager->storage_task_runner_.get(), FROM_HERE,
base::BindOnce(&ConversionStorage::GetConversionsToReport,
base::Unretained(manager->storage_.get()),
max_report_time),
base::BindOnce(base::BindLambdaForTesting(
[&](std::vector<ConversionReport> reports) {
conversion_reports = std::move(reports);
run_loop.Quit();
})));
run_loop.Run();
return conversion_reports;
}
} // namespace content