blob: ce2cd1c733c82c10c87afece315d0931c023df70 [file] [log] [blame]
// Copyright 2012 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/first_run/first_run.h"
#include <memory>
#include <string>
#include "base/base_switches.h"
#include "base/command_line.h"
#include "base/containers/contains.h"
#include "base/files/file_path.h"
#include "base/files/file_util.h"
#include "base/functional/bind.h"
#include "base/memory/ref_counted.h"
#include "base/numerics/safe_conversions.h"
#include "base/path_service.h"
#include "base/run_loop.h"
#include "base/strings/string_util.h"
#include "base/strings/utf_string_conversions.h"
#include "base/task/current_thread.h"
#include "base/test/metrics/histogram_tester.h"
#include "base/threading/thread_restrictions.h"
#include "build/build_config.h"
#include "chrome/browser/bookmarks/bookmark_model_factory.h"
#include "chrome/browser/browser_process.h"
#include "chrome/browser/extensions/component_loader.h"
#include "chrome/browser/first_run/first_run_internal.h"
#include "chrome/browser/history/history_service_factory.h"
#include "chrome/browser/importer/importer_list.h"
#include "chrome/browser/metrics/chrome_metrics_service_accessor.h"
#include "chrome/browser/prefs/chrome_pref_service_factory.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/browser/ui/tabs/tab_strip_model.h"
#include "chrome/common/channel_info.h"
#include "chrome/common/chrome_paths.h"
#include "chrome/common/chrome_switches.h"
#include "chrome/common/pref_names.h"
#include "chrome/common/url_constants.h"
#include "chrome/test/base/in_process_browser_test.h"
#include "chrome/test/base/ui_test_utils.h"
#include "components/bookmarks/browser/bookmark_model.h"
#include "components/bookmarks/browser/bookmark_node.h"
#include "components/history/content/browser/history_database_helper.h"
#include "components/history/core/browser/history_database_params.h"
#include "components/prefs/pref_service.h"
#include "components/user_prefs/user_prefs.h"
#include "components/variations/metrics.h"
#include "components/variations/pref_names.h"
#include "components/variations/variations_switches.h"
#include "content/public/browser/web_contents.h"
#include "content/public/common/content_switches.h"
#include "content/public/test/browser_test.h"
#include "content/public/test/browser_test_utils.h"
#include "content/public/test/test_launcher.h"
#include "testing/gtest/include/gtest/gtest.h"
typedef InProcessBrowserTest FirstRunBrowserTest;
namespace first_run {
#if !BUILDFLAG(IS_CHROMEOS)
namespace {
std::unique_ptr<KeyedService> BuildHistoryServiceAndRegisterCallback(
history::HistoryService::FaviconsChangedCallback callback,
base::CallbackListSubscription& favicon_changed_subscription,
content::BrowserContext* context) {
auto history_service = std::make_unique<history::HistoryService>();
history_service->Init(history::HistoryDatabaseParamsForPath(
context->GetPath(), version_info::Channel::UNKNOWN));
favicon_changed_subscription =
history_service->AddFaviconsChangedCallback(std::move(callback));
return history_service;
}
// A generic test class to be subclassed by test classes testing specific
// master_preferences. All subclasses must call SetInitialPreferencesForTest()
// from their SetUp() method before deferring the remainder of Setup() to this
// class.
class FirstRunMasterPrefsBrowserTestBase : public InProcessBrowserTest {
protected:
void SetUp() override {
// All users of this test class need to call SetInitialPreferencesForTest()
// before this class' SetUp() is invoked.
ASSERT_TRUE(text_.get());
ASSERT_TRUE(base::CreateTemporaryFile(&prefs_file_));
EXPECT_TRUE(base::WriteFile(prefs_file_, *text_));
SetInitialPrefsPathForTesting(prefs_file_);
// This invokes BrowserMain, and does the import, so must be done last.
InProcessBrowserTest::SetUp();
}
void TearDown() override {
EXPECT_TRUE(base::DeleteFile(prefs_file_));
InProcessBrowserTest::TearDown();
}
void SetUpCommandLine(base::CommandLine* command_line) override {
command_line->AppendSwitch(switches::kForceFirstRun);
EXPECT_EQ(AUTO_IMPORT_NONE, auto_import_state());
extensions::ComponentLoader::EnableBackgroundExtensionsForTesting();
}
#if BUILDFLAG(IS_MAC) || BUILDFLAG(IS_LINUX)
void SetUpInProcessBrowserTestFixture() override {
InProcessBrowserTest::SetUpInProcessBrowserTestFixture();
// Suppress first run dialog since it blocks test progress.
internal::ForceFirstRunDialogShownForTesting(false);
}
#endif
void SetInitialPreferencesForTest(const char text[]) {
text_ = std::make_unique<std::string>(text);
}
private:
base::FilePath prefs_file_;
std::unique_ptr<std::string> text_;
};
template <const char text[]>
class FirstRunMasterPrefsBrowserTestT
: public FirstRunMasterPrefsBrowserTestBase {
protected:
void SetUp() override {
SetInitialPreferencesForTest(text);
FirstRunMasterPrefsBrowserTestBase::SetUp();
}
};
// Returns the true expected import state, derived from the original
// |expected_import_state|, for the current test machine's configuration. Some
// bot configurations do not have another profile (browser) to import from and
// thus the import must not be expected to have occurred.
int MaskExpectedImportState(int expected_import_state) {
std::unique_ptr<ImporterList> importer_list(new ImporterList());
base::RunLoop run_loop;
importer_list->DetectSourceProfiles(
g_browser_process->GetApplicationLocale(),
false, // include_interactive_profiles?
run_loop.QuitClosure());
run_loop.Run();
int source_profile_count = importer_list->count();
#if BUILDFLAG(IS_WIN)
// On Windows, the importer's DetectIEProfiles() will always add to the count.
// Internet Explorer always exists and always has something to import.
EXPECT_GT(source_profile_count, 0);
#endif
if (source_profile_count == 0)
return expected_import_state & ~AUTO_IMPORT_PROFILE_IMPORTED;
return expected_import_state;
}
} // namespace
const char kImportDefault[] =
"{\n"
"}\n";
typedef FirstRunMasterPrefsBrowserTestT<kImportDefault>
FirstRunMasterPrefsImportDefault;
// No items are imported by default.
IN_PROC_BROWSER_TEST_F(FirstRunMasterPrefsImportDefault, ImportDefault) {
EXPECT_EQ(MaskExpectedImportState(AUTO_IMPORT_CALLED), auto_import_state());
}
const char kImportAll[] =
"{\n"
" \"distribution\": {\n"
" \"import_bookmarks\": true,\n"
" \"import_history\": true,\n"
" \"import_home_page\": true,\n"
" \"import_search_engine\": true\n"
" }\n"
"}\n";
typedef FirstRunMasterPrefsBrowserTestT<kImportAll>
FirstRunMasterPrefsImportAll;
IN_PROC_BROWSER_TEST_F(FirstRunMasterPrefsImportAll, ImportAll) {
EXPECT_EQ(MaskExpectedImportState(AUTO_IMPORT_CALLED |
AUTO_IMPORT_PROFILE_IMPORTED),
auto_import_state());
}
// The bookmarks file doesn't actually need to exist for this integration test
// to trigger the interaction being tested.
const char kImportBookmarksFile[] =
"{\n"
" \"distribution\": {\n"
" \"import_bookmarks_from_file\": \"/foo/doesntexists.wtv\"\n"
" }\n"
"}\n";
typedef FirstRunMasterPrefsBrowserTestT<kImportBookmarksFile>
FirstRunMasterPrefsImportBookmarksFile;
IN_PROC_BROWSER_TEST_F(FirstRunMasterPrefsImportBookmarksFile,
ImportBookmarksFile) {
EXPECT_EQ(MaskExpectedImportState(AUTO_IMPORT_CALLED |
AUTO_IMPORT_BOOKMARKS_FILE_IMPORTED),
auto_import_state());
}
// Test an import with all import options disabled. This is a regression test
// for http://crbug.com/169984 where this would cause the import process to
// stay running, and the NTP to be loaded with no apps.
const char kImportNothing[] =
"{\n"
" \"distribution\": {\n"
" \"import_bookmarks\": false,\n"
" \"import_history\": false,\n"
" \"import_home_page\": false,\n"
" \"import_search_engine\": false\n"
" }\n"
"}\n";
typedef FirstRunMasterPrefsBrowserTestT<kImportNothing>
FirstRunMasterPrefsImportNothing;
IN_PROC_BROWSER_TEST_F(FirstRunMasterPrefsImportNothing,
ImportNothingAndShowNewTabPage) {
EXPECT_EQ(AUTO_IMPORT_CALLED, auto_import_state());
ASSERT_TRUE(ui_test_utils::NavigateToURL(browser(),
GURL(chrome::kChromeUINewTabURL)));
content::WebContents* tab = browser()->tab_strip_model()->GetWebContentsAt(0);
EXPECT_TRUE(WaitForLoadStop(tab));
}
// Test first run with some tracked preferences.
const char kWithTrackedPrefs[] =
"{\n"
" \"homepage\": \"example.com\",\n"
" \"homepage_is_newtabpage\": false\n"
"}\n";
// A test fixture that will run in a first run scenario with master_preferences
// set to kWithTrackedPrefs.
class FirstRunMasterPrefsWithTrackedPreferences
: public FirstRunMasterPrefsBrowserTestT<kWithTrackedPrefs> {
protected:
void SetUpInProcessBrowserTestFixture() override {
FirstRunMasterPrefsBrowserTestT::SetUpInProcessBrowserTestFixture();
// Bots are on a domain, turn off the domain check for settings hardening in
// order to be able to test all SettingsEnforcement groups.
chrome_prefs::DisableDomainCheckForTesting();
}
};
IN_PROC_BROWSER_TEST_F(FirstRunMasterPrefsWithTrackedPreferences,
TrackedPreferencesSurviveFirstRun) {
const PrefService* user_prefs = browser()->profile()->GetPrefs();
EXPECT_EQ("example.com", user_prefs->GetString(prefs::kHomePage));
EXPECT_FALSE(user_prefs->GetBoolean(prefs::kHomePageIsNewTabPage));
// The test for kHomePageIsNewTabPage above relies on the fact that true is
// the default (hence false must be the user's pref); ensure this fact remains
// true.
const base::Value* default_homepage_is_ntp_value =
user_prefs->GetDefaultPrefValue(prefs::kHomePageIsNewTabPage);
ASSERT_TRUE(default_homepage_is_ntp_value->is_bool());
EXPECT_TRUE(default_homepage_is_ntp_value->GetBool());
}
#define COMPRESSED_SEED_TEST_VALUE \
"H4sIAAAAAAAA/+LSME4xsTAySjYzSDQ1S01KSk1KMUg1SjI1Tk4yMjI2NDMzTzEySjRPMxA6xs" \
"glH+rrqBual5mWX5SbWVKpG1KUmZija2igG5BalJyaV2LB6MXDxZFelF9aEG9gKIDMM0LhGaPw" \
"TFB4pig8MxSeOQrPAoVnKcDoxc3FnpKalliaUyLAGCSiwaDBqMGkwazBosGqwZbR8LppA38CIy" \
"AAAP//KpzsDPMAAAA="
#define SEED_SIGNATURE_TEST_VALUE \
"MEUCIQCo4D8Ad0pMlFoLT4mMrv7/ZK7PqyEmJlW5jciua6mluAIgQQKcj352r/sjq8b98W+jRk" \
"dwyDZn3ocgV01juWKx3u0="
constexpr char kCompressedSeedTestValue[] = COMPRESSED_SEED_TEST_VALUE;
constexpr char kSignatureValue[] = SEED_SIGNATURE_TEST_VALUE;
const char kWithVariationsPrefs[] =
"{\n"
" \"variations_compressed_seed\": \"" COMPRESSED_SEED_TEST_VALUE
"\",\n"
" \"variations_seed_signature\": \"" SEED_SIGNATURE_TEST_VALUE
"\"\n"
"}\n";
#undef COMPRESSED_SEED_TEST_VALUE
#undef SEED_SIGNATURE_TEST_VALUE
// Note: This test is parametrized on metrics consent state, since that affects
// field trial randomization.
class FirstRunMasterPrefsVariationsSeedTest
: public FirstRunMasterPrefsBrowserTestT<kWithVariationsPrefs>,
public testing::WithParamInterface<bool> {
public:
FirstRunMasterPrefsVariationsSeedTest() : metrics_consent_(GetParam()) {
base::CommandLine::ForCurrentProcess()->AppendSwitch(
variations::switches::kDisableFieldTrialTestingConfig);
}
FirstRunMasterPrefsVariationsSeedTest(
const FirstRunMasterPrefsVariationsSeedTest&) = delete;
FirstRunMasterPrefsVariationsSeedTest& operator=(
const FirstRunMasterPrefsVariationsSeedTest&) = delete;
~FirstRunMasterPrefsVariationsSeedTest() override = default;
void SetUp() override {
// Make metrics reporting work same as in Chrome branded builds, for test
// consistency between Chromium and Chrome builds.
ChromeMetricsServiceAccessor::SetForceIsMetricsReportingEnabledPrefLookup(
true);
// Based on GetParam(), either enable or disable metrics reporting.
ChromeMetricsServiceAccessor::SetMetricsAndCrashReportingForTesting(
&metrics_consent_);
FirstRunMasterPrefsBrowserTestT::SetUp();
}
// Writes the trial group to the temporary file.
void WriteTrialGroupToTestFile(const std::string& trial_group) {
base::ScopedAllowBlockingForTesting allow_blocking;
EXPECT_TRUE(base::WriteFile(GetTestFilePath(), trial_group));
}
// Reads the trial group from the temporary file.
std::string ReadTrialGroupFromTestFile() {
base::ScopedAllowBlockingForTesting allow_blocking;
char data[256];
int bytes_read = base::ReadFile(GetTestFilePath(), data, sizeof(data));
EXPECT_NE(-1, bytes_read);
return std::string(data, bytes_read);
}
protected:
base::HistogramTester histogram_tester_;
private:
const bool metrics_consent_;
// Returns a file path for persisting a field trial's state. The path is
// under the user data dir, so should only persist between pairs of PRE_Foo
// and Foo tests.
base::FilePath GetTestFilePath() {
base::FilePath user_data_dir;
EXPECT_TRUE(base::PathService::Get(chrome::DIR_USER_DATA, &user_data_dir));
return user_data_dir.AppendASCII("FirstRunMasterPrefsVariationsSeedTest");
}
};
IN_PROC_BROWSER_TEST_P(FirstRunMasterPrefsVariationsSeedTest, Test) {
// Tests variation migration from master_preferences to local_state.
EXPECT_EQ(kCompressedSeedTestValue,
g_browser_process->local_state()->GetString(
variations::prefs::kVariationsCompressedSeed));
EXPECT_EQ(kSignatureValue, g_browser_process->local_state()->GetString(
variations::prefs::kVariationsSeedSignature));
// Verify variations loaded in VariationsService by metrics.
histogram_tester_.ExpectUniqueSample(
"Variations.SeedLoadResult", variations::StoreSeedResult::kSuccess, 1);
histogram_tester_.ExpectUniqueSample(
"Variations.LoadSeedSignature",
variations::VerifySignatureResult::VALID_SIGNATURE, 1);
}
// The following tests are only enabled on Windows, since it is the only
// platform where master prefs is used to deliver first run variations. The
// tests do not pass on other platforms due to the provisional client id logic
// in metrics_state_manager.cc. See the comment there for details.
#if BUILDFLAG(IS_WIN)
// The trial and groups encoded in the above seed.
constexpr char kTrialName[] = "UMA-Uniformity-Trial-10-Percent";
const char* kTrialGroups[] = {"default", "group_01", "group_02", "group_03",
"group_04", "group_05", "group_06", "group_07",
"group_08", "group_09"};
IN_PROC_BROWSER_TEST_P(FirstRunMasterPrefsVariationsSeedTest, PRE_SecondRun) {
// Check that the trial from the seed exists and is in one of the expected
// states. Persist the state so that we can verify its randomization persists
// in FirstRunMasterPrefsVariationsSeedTest.SecondRun.
const std::string group_name = base::FieldTrialList::FindFullName(kTrialName);
ASSERT_TRUE(base::Contains(kTrialGroups, group_name)) << group_name;
// Ensure trial is active (not disabled).
ASSERT_TRUE(base::FieldTrialList::IsTrialActive(kTrialName));
WriteTrialGroupToTestFile(group_name);
}
IN_PROC_BROWSER_TEST_P(FirstRunMasterPrefsVariationsSeedTest, SecondRun) {
// This test runs after PRE_SecondRun and verifies that the trial state on
// the second run matches what was seen in the PRE_ test.
const std::string group_name = base::FieldTrialList::FindFullName(kTrialName);
ASSERT_TRUE(base::Contains(kTrialGroups, group_name)) << group_name;
// Ensure trial is active (not disabled).
ASSERT_TRUE(base::FieldTrialList::IsTrialActive(kTrialName));
// Read the trial group name that was saved by PRE_ForceTrials from the
// corresponding test file.
EXPECT_EQ(group_name, ReadTrialGroupFromTestFile());
}
#endif // BUILDFLAG(IS_WIN)
INSTANTIATE_TEST_SUITE_P(FirstRunMasterPrefsVariationsSeedTests,
FirstRunMasterPrefsVariationsSeedTest,
testing::Bool());
struct FirstRunMasterPrefsImportBookmarkFaviconBrowserTestParams {
const char* initial_prefs;
bool valid;
const char* test_name;
};
class FirstRunMasterPrefsImportBookmarkFaviconBrowserTest
: public FirstRunMasterPrefsBrowserTestBase,
public ::testing::WithParamInterface<
FirstRunMasterPrefsImportBookmarkFaviconBrowserTestParams> {
protected:
void SetUp() override {
SetInitialPreferencesForTest(GetParam().initial_prefs);
FirstRunMasterPrefsBrowserTestBase::SetUp();
}
void SetUpBrowserContextKeyedServices(
content::BrowserContext* context) override {
InProcessBrowserTest::SetUpBrowserContextKeyedServices(context);
history::HistoryService::FaviconsChangedCallback cb = base::BindRepeating(
&FirstRunMasterPrefsImportBookmarkFaviconBrowserTest::OnFaviconChanged,
base::Unretained(this));
HistoryServiceFactory::GetInstance()->SetTestingFactory(
context, base::BindRepeating(&BuildHistoryServiceAndRegisterCallback,
std::move(cb),
std::ref(favicon_changed_subscription_)));
}
void OnFaviconChanged(const std::set<GURL>& page_urls, const GURL& icon_url) {
if (page_urls.contains(page_url_)) {
favicon_updated_ = true;
}
}
bool favicon_updated_ = false;
const GURL page_url_ = GURL("https://www.abcd1234.com");
base::CallbackListSubscription favicon_changed_subscription_;
};
IN_PROC_BROWSER_TEST_P(FirstRunMasterPrefsImportBookmarkFaviconBrowserTest,
ImportBookmarksDict) {
Profile* profile = browser()->profile();
bookmarks::BookmarkModel* bookmark_model =
BookmarkModelFactory::GetForBrowserContext(profile);
const bookmarks::BookmarkNode* bar = bookmark_model->bookmark_bar_node();
ASSERT_EQ(1u, bar->children().size());
const bookmarks::BookmarkNode* node1 = bar->children()[0].get();
EXPECT_TRUE(node1->is_url());
EXPECT_EQ(u"ABCD1234", node1->GetTitle());
EXPECT_EQ(page_url_, node1->url());
if (GetParam().valid) {
EXPECT_TRUE(base::test::RunUntil([this]() { return favicon_updated_; }));
} else {
base::RunLoop().RunUntilIdle();
EXPECT_FALSE(favicon_updated_);
}
}
const char kImportBookmarksDictValidFaviconData[] = R"({
"bookmarks": {
"first_run_bookmarks": {
"children": [
{
"name": "ABCD1234",
"type": "url",
"url": "https://www.abcd1234.com",
"icon_data_url": "
AgCAYAAABzenr0AAAACXBIWXMAAAsSAAALEgHS3X78AAAE3ElEQVRYhc2XW2xUVRSGv30u
U9rTMhUaCuVixQSMpLYmPIhYwAcTIyKFROI1KQmkxBgj8UEf5MEYiSYmPvhAfCASY02MUa
FEHjSRQYtGNDAqIhKxQJVeodPLMJ2Zc/by4cz91tZ4+5Od2Wf2Xuv/z9lrr722EhFmiTag
A9gEbCwz5wQQAg4D4Vl5FZGZWqeIhGXuCKdsK/qvNNgsIqG/QFyIUMrXnAR0iEjkbyBPI5
LyWcSlpDgGOoG3S67XV71wshd+CMPwEAiAAShoXIy0tsL6dtS6u8qt+E7gUO4fhQI6gI+L
zM6ehjf2w9AQJC1QgCi/KRMwQZmIYSLKRBYvwXj2KVTLmlIituEHaZGAZvzIDeZN/+BVOP
whTAX850QgS44CZWWaoBDTRgzbd77lfoxdTxYKGMffUZfA/35pHCoif+8F+Pxdvx/wSr1N
PpSR/3z2HERvFM4KkrMMaYtOCvf2p8/A5W5ouAHz4+AkoDoJpguGB6QEpT6GAKJUVsstKz
Be2QdOTSmpG1OcmSUIA62Z4eEeCD0BV+pgnoahOhhwwBCwVsL23bCuHZxaf340Cl9/g+5+
H0YjcOvKSuRpfA+0KRFpA87kDX2xCgYHIWpDNOALmLLhju2w4/kscSGiUeTIMdTWzTORp3
GnhR/5WYwdhUA/NBhQKzDtQZUHddvhkZcru3Mc1GMPz4Y4jQ4LP7dnETkK8wywACVgu2DU
wQOvzcXxbLHJojD4pnuz/WoBF1j2KATyN0ga310p41pAkc0xCqEpqFhcr3JnbbSK7OJ9oA
yUpcAEAsDSLWVfYXd3OQGFGVaxpx262vP/Ldi4KelCZk+LoWDeirICykKpbEs/l0CRABGd
7aeMhNLGcxJTxk/xFxAN4uJpl6R4TIuLG79U0b+gEbxU06TSUqpVRlEMEGhGJy4zJRpLmV
zXHtWRIzTUbSjpoOueNInK++35EQbGZ+THwi+jMjtB17bjXesDDKLaT7f9I+9Q37QPy6wv
crCnXeWQ+5ichu5v08IkM7725iLzEwZ+DZeB1G9lVAv9nseg1vziac7EJ/nk1z0zv04KB8
9EGXVjRO0pkkYCVyVJGgnWFsdyyCDnbAaw6zsYtldwUQtXtTCshUEx+HL0Mw5e2MuUO1mR
/MBPf7A//DvjNcNMVI8QC0zhGR4PtZQM5MMlD6OJiRDHz91LRBTjYtAnDtd1DWPikAi0sq
u5i/sa1hO0nIynnoGLvHn+Ar0DMQKx5VhuNbZbg5MIYno2x3fXsTSYJyJzGEGJMizUt5fj
Vw8wLjaXJciEOEyKw5DXSJwASapI6lpqzflEkh6SDEJ8CYiJii3HdB0CroMdX0BXy0Je2l
AUPzuBQ7kVUYiCtNx9oYuPhnoYkIV4YjGmb2JSGnBReFKFxsbVVSgE8Rz09BJQGuJNWLEm
PMOlrXYZJ7fdXkh+gtQZlJsHOvHLpQweX/UWmxp3ABCXQHZAzEzXUG6mr6yJTN+1x3lw+S
KObV5dSD6e4vJtZlOUnoqc5rmfX+e32AQxCSIYCCag0NryT01dhXjVSHIBtbKUF29bx9Or
VxW6ggpFaRqdlCnLj42c4uhImN6x8/RPXwMUIgYKYb5Zz93BNrY0ruHBRS0E7apSLorK8v
/8YvK/vZr9a5fTUjFQDv/I9fxPxUx0d1WRkbMAAAAASUVORK5CYII="
}
]
}
}
})";
constexpr char kImportBookmarksDictNoFaviconData[] = R"({
"bookmarks": {
"first_run_bookmarks": {
"children": [
{
"name": "ABCD1234",
"type": "url",
"url": "https://www.abcd1234.com"
}
]
}
}
})";
constexpr char kImportBookmarksDictMalformedFaviconData[] = R"({
"bookmarks": {
"first_run_bookmarks": {
"children": [
{
"name": "ABCD1234",
"type": "url",
"url": "https://www.abcd1234.com",
"icon_data_url": ""
}
]
}
}
})";
constexpr char kImportBookmarksDictBadFaviconUrlScheme[] = R"({
"bookmarks": {
"first_run_bookmarks": {
"children": [
{
"name": "ABCD1234",
"type": "url",
"url": "https://www.abcd1234.com",
"icon_data_url": "badscheme;base64,empty"
}
]
}
}
})";
INSTANTIATE_TEST_SUITE_P(
,
FirstRunMasterPrefsImportBookmarkFaviconBrowserTest,
::testing::Values(
FirstRunMasterPrefsImportBookmarkFaviconBrowserTestParams{
.initial_prefs = kImportBookmarksDictValidFaviconData,
.valid = true,
.test_name = "ValidFaviconData",
},
FirstRunMasterPrefsImportBookmarkFaviconBrowserTestParams{
.initial_prefs = kImportBookmarksDictNoFaviconData,
.valid = false,
.test_name = "NoFaviconData",
},
FirstRunMasterPrefsImportBookmarkFaviconBrowserTestParams{
.initial_prefs = kImportBookmarksDictMalformedFaviconData,
.valid = false,
.test_name = "MalformedFaviconData",
},
FirstRunMasterPrefsImportBookmarkFaviconBrowserTestParams{
.initial_prefs = kImportBookmarksDictBadFaviconUrlScheme,
.valid = false,
.test_name = "BadFaviconUrlScheme",
}),
[](const ::testing::TestParamInfo<
FirstRunMasterPrefsImportBookmarkFaviconBrowserTest::ParamType>& info) {
return info.param.test_name;
});
#endif // !BUILDFLAG(IS_CHROMEOS)
} // namespace first_run