| // 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 <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/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/test/metrics/histogram_tester.h" |
| #include "base/threading/thread_restrictions.h" |
| #include "build/build_config.h" |
| #include "chrome/browser/browser_process.h" |
| #include "chrome/browser/extensions/component_loader.h" |
| #include "chrome/browser/first_run/first_run.h" |
| #include "chrome/browser/first_run/first_run_internal.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/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/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 { |
| |
| // 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()); |
| |
| #endif // !BUILDFLAG(IS_CHROMEOS) |
| |
| } // namespace first_run |