|  | // 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/themes/theme_syncable_service.h" | 
|  |  | 
|  | #include <memory> | 
|  |  | 
|  | #include "base/bind.h" | 
|  | #include "base/command_line.h" | 
|  | #include "base/compiler_specific.h" | 
|  | #include "base/files/file_path.h" | 
|  | #include "base/memory/ptr_util.h" | 
|  | #include "base/memory/raw_ptr.h" | 
|  | #include "base/no_destructor.h" | 
|  | #include "base/run_loop.h" | 
|  | #include "base/time/time.h" | 
|  | #include "base/values.h" | 
|  | #include "build/build_config.h" | 
|  | #include "build/chromeos_buildflags.h" | 
|  | #include "chrome/browser/extensions/extension_service.h" | 
|  | #include "chrome/browser/extensions/test_extension_system.h" | 
|  | #include "chrome/browser/profiles/profile.h" | 
|  | #include "chrome/browser/themes/theme_helper.h" | 
|  | #include "chrome/browser/themes/theme_service.h" | 
|  | #include "chrome/browser/themes/theme_service_factory.h" | 
|  | #include "chrome/common/extensions/extension_test_util.h" | 
|  | #include "chrome/test/base/testing_profile.h" | 
|  | #include "components/sync/base/client_tag_hash.h" | 
|  | #include "components/sync/model/sync_error.h" | 
|  | #include "components/sync/protocol/entity_specifics.pb.h" | 
|  | #include "components/sync/protocol/theme_specifics.pb.h" | 
|  | #include "components/sync/test/fake_sync_change_processor.h" | 
|  | #include "components/sync/test/sync_change_processor_wrapper_for_test.h" | 
|  | #include "components/sync/test/sync_error_factory_mock.h" | 
|  | #include "content/public/test/browser_task_environment.h" | 
|  | #include "extensions/browser/extension_prefs.h" | 
|  | #include "extensions/browser/extension_registry.h" | 
|  | #include "extensions/common/extension.h" | 
|  | #include "extensions/common/manifest_constants.h" | 
|  | #include "extensions/common/manifest_url_handlers.h" | 
|  | #include "extensions/common/permissions/api_permission_set.h" | 
|  | #include "extensions/common/permissions/permission_set.h" | 
|  | #include "testing/gtest/include/gtest/gtest.h" | 
|  | #include "third_party/abseil-cpp/absl/types/optional.h" | 
|  |  | 
|  | #if BUILDFLAG(IS_CHROMEOS_ASH) | 
|  | #include "chrome/browser/ash/login/users/scoped_test_user_manager.h" | 
|  | #include "chrome/browser/ash/settings/scoped_cros_settings_test_helper.h" | 
|  | #endif | 
|  |  | 
|  | using std::string; | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | static const char kCustomThemeName[] = "name"; | 
|  | static const char kCustomThemeUrl[] = "http://update.url/foo"; | 
|  |  | 
|  | #if BUILDFLAG(IS_WIN) | 
|  | const base::FilePath::CharType kExtensionFilePath[] = | 
|  | FILE_PATH_LITERAL("c:\\foo"); | 
|  | #elif BUILDFLAG(IS_POSIX) || BUILDFLAG(IS_FUCHSIA) | 
|  | const base::FilePath::CharType kExtensionFilePath[] = FILE_PATH_LITERAL("/oo"); | 
|  | #else | 
|  | #error "Unknown platform" | 
|  | #endif | 
|  |  | 
|  | const ThemeHelper& GetThemeHelper() { | 
|  | static base::NoDestructor<std::unique_ptr<ThemeHelper>> theme_helper( | 
|  | std::make_unique<ThemeHelper>()); | 
|  | return **theme_helper; | 
|  | } | 
|  |  | 
|  | class FakeThemeService : public ThemeService { | 
|  | public: | 
|  | FakeThemeService() : ThemeService(nullptr, GetThemeHelper()) {} | 
|  |  | 
|  | // ThemeService implementation | 
|  | void DoSetTheme(const extensions::Extension* extension, | 
|  | bool suppress_infobar) override { | 
|  | is_dirty_ = true; | 
|  | theme_extension_ = extension; | 
|  | using_system_theme_ = false; | 
|  | using_default_theme_ = false; | 
|  | using_policy_theme_ = false; | 
|  | might_show_infobar_ = !suppress_infobar; | 
|  | color_ = 0; | 
|  | } | 
|  |  | 
|  | void BuildAutogeneratedThemeFromColor(SkColor color) override { | 
|  | is_dirty_ = true; | 
|  | color_ = color; | 
|  | theme_extension_.reset(); | 
|  | using_system_theme_ = false; | 
|  | using_default_theme_ = false; | 
|  | using_policy_theme_ = false; | 
|  | } | 
|  |  | 
|  | void UseTheme(ui::SystemTheme system_theme) override { | 
|  | if (system_theme == ui::SystemTheme::kDefault) | 
|  | UseDefaultTheme(); | 
|  | else | 
|  | UseSystemTheme(); | 
|  | } | 
|  |  | 
|  | void UseDefaultTheme() override { | 
|  | is_dirty_ = true; | 
|  | using_default_theme_ = true; | 
|  | using_system_theme_ = false; | 
|  | using_policy_theme_ = false; | 
|  | theme_extension_.reset(); | 
|  | color_ = 0; | 
|  | } | 
|  |  | 
|  | void UseSystemTheme() override { | 
|  | is_dirty_ = true; | 
|  | using_system_theme_ = true; | 
|  | using_default_theme_ = false; | 
|  | using_policy_theme_ = false; | 
|  | theme_extension_.reset(); | 
|  | color_ = 0; | 
|  | } | 
|  |  | 
|  | void BuildAutogeneratedPolicyTheme() override { | 
|  | is_dirty_ = true; | 
|  | color_ = SkColorSetRGB(100, 100, 0); | 
|  | theme_extension_.reset(); | 
|  | using_system_theme_ = false; | 
|  | using_default_theme_ = false; | 
|  | using_policy_theme_ = true; | 
|  | } | 
|  |  | 
|  | bool IsSystemThemeDistinctFromDefaultTheme() const override { | 
|  | return distinct_from_default_theme_; | 
|  | } | 
|  |  | 
|  | void set_distinct_from_default_theme(bool is_distinct) { | 
|  | distinct_from_default_theme_ = is_distinct; | 
|  | } | 
|  |  | 
|  | bool UsingDefaultTheme() const override { return using_default_theme_; } | 
|  |  | 
|  | bool UsingSystemTheme() const override { return using_system_theme_; } | 
|  |  | 
|  | bool UsingExtensionTheme() const override { return !!theme_extension_; } | 
|  |  | 
|  | bool UsingAutogeneratedTheme() const override { return color_ != 0; } | 
|  |  | 
|  | bool UsingPolicyTheme() const override { return using_policy_theme_; } | 
|  |  | 
|  | string GetThemeID() const override { | 
|  | return UsingExtensionTheme() ? theme_extension_->id() : std::string(); | 
|  | } | 
|  |  | 
|  | SkColor GetAutogeneratedThemeColor() const override { return color_; } | 
|  |  | 
|  | const extensions::Extension* theme_extension() const { | 
|  | return theme_extension_.get(); | 
|  | } | 
|  |  | 
|  | bool is_dirty() const { return is_dirty_; } | 
|  |  | 
|  | void MarkClean() { is_dirty_ = false; } | 
|  |  | 
|  | bool might_show_infobar() const { return might_show_infobar_; } | 
|  |  | 
|  | private: | 
|  | bool using_system_theme_ = false; | 
|  | bool using_default_theme_ = false; | 
|  | bool using_policy_theme_ = false; | 
|  | bool distinct_from_default_theme_ = false; | 
|  | scoped_refptr<const extensions::Extension> theme_extension_; | 
|  | bool is_dirty_ = false; | 
|  | bool might_show_infobar_ = false; | 
|  | SkColor color_; | 
|  | }; | 
|  |  | 
|  | std::unique_ptr<KeyedService> BuildMockThemeService( | 
|  | content::BrowserContext* profile) { | 
|  | return std::make_unique<FakeThemeService>(); | 
|  | } | 
|  |  | 
|  | scoped_refptr<extensions::Extension> MakeThemeExtension( | 
|  | const base::FilePath& extension_path, | 
|  | const string& name, | 
|  | extensions::mojom::ManifestLocation location, | 
|  | const string& update_url) { | 
|  | base::Value::Dict source; | 
|  | source.Set(extensions::manifest_keys::kName, name); | 
|  | source.Set(extensions::manifest_keys::kTheme, base::Value::Dict()); | 
|  | source.Set(extensions::manifest_keys::kUpdateURL, update_url); | 
|  | source.Set(extensions::manifest_keys::kVersion, "0.0.0.0"); | 
|  | string error; | 
|  | scoped_refptr<extensions::Extension> extension = | 
|  | extensions::Extension::Create( | 
|  | extension_path, location, source, | 
|  | extensions::Extension::NO_FLAGS, &error); | 
|  | EXPECT_TRUE(extension.get()); | 
|  | EXPECT_EQ("", error); | 
|  | return extension; | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | class ThemeSyncableServiceTest : public testing::Test, | 
|  | public ThemeSyncableService::Observer { | 
|  | protected: | 
|  | ThemeSyncableServiceTest() : fake_theme_service_(nullptr) {} | 
|  |  | 
|  | ~ThemeSyncableServiceTest() override {} | 
|  |  | 
|  | void SetUp() override { | 
|  | // Setting a matching update URL is necessary to make the test theme | 
|  | // considered syncable. | 
|  | extension_test_util::SetGalleryUpdateURL(GURL(kCustomThemeUrl)); | 
|  |  | 
|  | profile_ = std::make_unique<TestingProfile>(); | 
|  | fake_theme_service_ = BuildForProfile(profile_.get()); | 
|  | theme_sync_service_ = std::make_unique<ThemeSyncableService>( | 
|  | profile_.get(), fake_theme_service_); | 
|  | theme_sync_service_->AddObserver(this); | 
|  | fake_change_processor_ = | 
|  | std::make_unique<syncer::FakeSyncChangeProcessor>(); | 
|  | SetUpExtension(); | 
|  | } | 
|  |  | 
|  | void TearDown() override { | 
|  | theme_sync_service_.reset(); | 
|  | profile_.reset(); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  | } | 
|  |  | 
|  | void SetUpExtension() { | 
|  | base::CommandLine command_line(base::CommandLine::NO_PROGRAM); | 
|  | extensions::TestExtensionSystem* test_ext_system = | 
|  | static_cast<extensions::TestExtensionSystem*>( | 
|  | extensions::ExtensionSystem::Get(profile_.get())); | 
|  | extensions::ExtensionService* service = | 
|  | test_ext_system->CreateExtensionService( | 
|  | &command_line, base::FilePath(kExtensionFilePath), false); | 
|  | EXPECT_TRUE(service->extensions_enabled()); | 
|  | service->Init(); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | // Create and add custom theme extension so the ThemeSyncableService can | 
|  | // find it. | 
|  | theme_extension_ = MakeThemeExtension(base::FilePath(kExtensionFilePath), | 
|  | kCustomThemeName, | 
|  | GetThemeLocation(), | 
|  | kCustomThemeUrl); | 
|  | extensions::ExtensionPrefs::Get(profile_.get()) | 
|  | ->AddGrantedPermissions(theme_extension_->id(), | 
|  | extensions::PermissionSet()); | 
|  | service->AddExtension(theme_extension_.get()); | 
|  | extensions::ExtensionRegistry* registry = | 
|  | extensions::ExtensionRegistry::Get(profile_.get()); | 
|  | ASSERT_EQ(1u, registry->enabled_extensions().size()); | 
|  | } | 
|  |  | 
|  | // Overridden in PolicyInstalledThemeTest below. | 
|  | virtual extensions::mojom::ManifestLocation GetThemeLocation() { | 
|  | return extensions::mojom::ManifestLocation::kInternal; | 
|  | } | 
|  |  | 
|  | FakeThemeService* BuildForProfile(Profile* profile) { | 
|  | return static_cast<FakeThemeService*>( | 
|  | ThemeServiceFactory::GetInstance()->SetTestingFactoryAndUse( | 
|  | profile, base::BindRepeating(&BuildMockThemeService))); | 
|  | } | 
|  |  | 
|  | syncer::SyncDataList MakeThemeDataList( | 
|  | const sync_pb::ThemeSpecifics& theme_specifics) { | 
|  | syncer::SyncDataList list; | 
|  | sync_pb::EntitySpecifics entity_specifics; | 
|  | entity_specifics.mutable_theme()->CopyFrom(theme_specifics); | 
|  | list.push_back(syncer::SyncData::CreateLocalData( | 
|  | ThemeSyncableService::kSyncEntityClientTag, | 
|  | ThemeSyncableService::kSyncEntityTitle, entity_specifics)); | 
|  | return list; | 
|  | } | 
|  |  | 
|  | void OnThemeSyncStarted(ThemeSyncableService::ThemeSyncState state) override { | 
|  | state_ = state; | 
|  | } | 
|  |  | 
|  | bool HasThemeSyncStarted() { return state_ != absl::nullopt; } | 
|  |  | 
|  | bool HasThemeSyncTriggeredExtensionInstallation() { | 
|  | return state_ && *state_ == ThemeSyncableService::ThemeSyncState:: | 
|  | kWaitingForExtensionInstallation; | 
|  | } | 
|  |  | 
|  | // Needed for setting up extension service. | 
|  | content::BrowserTaskEnvironment task_environment_; | 
|  |  | 
|  | #if BUILDFLAG(IS_CHROMEOS_ASH) | 
|  | ash::ScopedCrosSettingsTestHelper cros_settings_test_helper_; | 
|  | ash::ScopedTestUserManager test_user_manager_; | 
|  | #endif | 
|  |  | 
|  | std::unique_ptr<TestingProfile> profile_; | 
|  | raw_ptr<FakeThemeService> fake_theme_service_; | 
|  | scoped_refptr<extensions::Extension> theme_extension_; | 
|  | std::unique_ptr<ThemeSyncableService> theme_sync_service_; | 
|  | std::unique_ptr<syncer::FakeSyncChangeProcessor> fake_change_processor_; | 
|  | absl::optional<ThemeSyncableService::ThemeSyncState> state_; | 
|  | }; | 
|  |  | 
|  | class PolicyInstalledThemeTest : public ThemeSyncableServiceTest { | 
|  | extensions::mojom::ManifestLocation GetThemeLocation() override { | 
|  | return extensions::mojom::ManifestLocation::kExternalPolicyDownload; | 
|  | } | 
|  | }; | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, AreThemeSpecificsEqual) { | 
|  | sync_pb::ThemeSpecifics a, b; | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | // Custom vs. non-custom. | 
|  |  | 
|  | a.set_use_custom_theme(true); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | a.set_use_custom_theme(false); | 
|  | b.set_use_custom_theme(true); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | // Custom theme equality for extensions. | 
|  | a.set_use_custom_theme(true); | 
|  | b.set_use_custom_theme(true); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | a.set_custom_theme_id("id"); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | b.set_custom_theme_id("id"); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | a.set_custom_theme_update_url("http://update.url"); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | a.set_custom_theme_name("name"); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | // Theme equality for autogenerated. | 
|  | a.set_use_custom_theme(true); | 
|  | b.set_use_custom_theme(false); | 
|  | b.mutable_autogenerated_theme()->set_color(SkColorSetRGB(0, 0, 0)); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | b.set_use_custom_theme(true); | 
|  | b.clear_autogenerated_theme(); | 
|  | a.set_use_custom_theme(false); | 
|  | a.mutable_autogenerated_theme()->set_color(SkColorSetRGB(0, 0, 0)); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | b.set_use_custom_theme(false); | 
|  | b.mutable_autogenerated_theme()->set_color(SkColorSetRGB(0, 0, 100)); | 
|  | a.mutable_autogenerated_theme()->set_color(SkColorSetRGB(0, 0, 100)); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | b.mutable_autogenerated_theme()->set_color(SkColorSetRGB(0, 0, 200)); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | // No theme equality. | 
|  |  | 
|  | a.clear_autogenerated_theme(); | 
|  | b.clear_autogenerated_theme(); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | a.set_use_system_theme_by_default(true); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_FALSE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  |  | 
|  | b.set_use_system_theme_by_default(true); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, false)); | 
|  | EXPECT_TRUE(ThemeSyncableService::AreThemeSpecificsEqual(a, b, true)); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, SetCurrentThemeDefaultTheme) { | 
|  | // Set up theme service to use custom theme. | 
|  | fake_theme_service_->SetTheme(theme_extension_.get()); | 
|  |  | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(sync_pb::ThemeSpecifics()), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_TRUE(HasThemeSyncStarted()); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | EXPECT_FALSE(fake_theme_service_->UsingDefaultTheme()); | 
|  | EXPECT_EQ(fake_theme_service_->theme_extension(), theme_extension_.get()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, SetCurrentThemeSystemTheme) { | 
|  | sync_pb::ThemeSpecifics theme_specifics; | 
|  | theme_specifics.set_use_system_theme_by_default(true); | 
|  |  | 
|  | // Set up theme service to use custom theme. | 
|  | fake_theme_service_->SetTheme(theme_extension_.get()); | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(theme_specifics), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_TRUE(HasThemeSyncStarted()); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | EXPECT_FALSE(fake_theme_service_->UsingSystemTheme()); | 
|  | EXPECT_EQ(fake_theme_service_->theme_extension(), theme_extension_.get()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, SetCurrentThemeCustomTheme_Extension) { | 
|  | sync_pb::ThemeSpecifics theme_specifics; | 
|  | theme_specifics.set_use_custom_theme(true); | 
|  | theme_specifics.set_custom_theme_id(theme_extension_->id()); | 
|  | theme_specifics.set_custom_theme_name(kCustomThemeName); | 
|  | theme_specifics.set_custom_theme_update_url(kCustomThemeUrl); | 
|  |  | 
|  | // Set up theme service to use default theme. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(theme_specifics), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_TRUE(HasThemeSyncStarted()); | 
|  | EXPECT_FALSE(HasThemeSyncTriggeredExtensionInstallation()); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | EXPECT_EQ(fake_theme_service_->theme_extension(), theme_extension_.get()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, SetCurrentThemeCustomTheme_Extension_Install) { | 
|  | sync_pb::ThemeSpecifics theme_specifics; | 
|  | theme_specifics.set_use_custom_theme(true); | 
|  | // Use an arbitrary id (such an extension is not installed, yet). | 
|  | theme_specifics.set_custom_theme_id("fake_extension_id"); | 
|  | theme_specifics.set_custom_theme_name(kCustomThemeName); | 
|  | theme_specifics.set_custom_theme_update_url(kCustomThemeUrl); | 
|  |  | 
|  | // Set up theme service to use default theme. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(theme_specifics), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_TRUE(HasThemeSyncStarted()); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | // The theme is not installed yet and thus, the default theme is still used. | 
|  | EXPECT_TRUE(fake_theme_service_->UsingDefaultTheme()); | 
|  | EXPECT_TRUE(HasThemeSyncTriggeredExtensionInstallation()); | 
|  | EXPECT_TRUE(extensions::ExtensionSystem::Get(profile_.get()) | 
|  | ->extension_service() | 
|  | ->pending_extension_manager() | 
|  | ->HasPendingExtensionFromSync()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, SetCurrentThemeCustomTheme_Autogenerated) { | 
|  | sync_pb::ThemeSpecifics theme_specifics; | 
|  | theme_specifics.set_use_custom_theme(false); | 
|  | theme_specifics.mutable_autogenerated_theme()->set_color( | 
|  | SkColorSetRGB(0, 0, 100)); | 
|  |  | 
|  | // Set up theme service to use default theme. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(theme_specifics), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_TRUE(HasThemeSyncStarted()); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | EXPECT_EQ(fake_theme_service_->GetAutogeneratedThemeColor(), | 
|  | SkColorSetRGB(0, 0, 100)); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, DontResetThemeWhenSpecificsAreEqual) { | 
|  | // Set up theme service to use default theme and expect no changes. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  | fake_theme_service_->MarkClean(); | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(sync_pb::ThemeSpecifics()), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_TRUE(HasThemeSyncStarted()); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | EXPECT_FALSE(fake_theme_service_->is_dirty()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, UpdateThemeSpecifics_CurrentTheme_Extension) { | 
|  | // Set up theme service to use custom theme. | 
|  | fake_theme_service_->SetTheme(theme_extension_.get()); | 
|  |  | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, syncer::SyncDataList(), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | const syncer::SyncChangeList& changes = fake_change_processor_->changes(); | 
|  | ASSERT_EQ(1u, changes.size()); | 
|  | EXPECT_EQ(syncer::SyncChange::ACTION_ADD, changes[0].change_type()); | 
|  | EXPECT_EQ(syncer::THEMES, changes[0].sync_data().GetDataType()); | 
|  |  | 
|  | const sync_pb::ThemeSpecifics& theme_specifics = | 
|  | changes[0].sync_data().GetSpecifics().theme(); | 
|  | EXPECT_TRUE(theme_specifics.use_custom_theme()); | 
|  | EXPECT_EQ(theme_extension_->id(), theme_specifics.custom_theme_id()); | 
|  | EXPECT_EQ(theme_extension_->name(), theme_specifics.custom_theme_name()); | 
|  | EXPECT_EQ( | 
|  | extensions::ManifestURL::GetUpdateURL(theme_extension_.get()).spec(), | 
|  | theme_specifics.custom_theme_update_url()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, | 
|  | UpdateThemeSpecifics_CurrentTheme_Autogenerated) { | 
|  | // Set up theme service to use autogenerated theme. | 
|  | fake_theme_service_->BuildAutogeneratedThemeFromColor( | 
|  | SkColorSetRGB(0, 0, 100)); | 
|  |  | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, syncer::SyncDataList(), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | const syncer::SyncChangeList& changes = fake_change_processor_->changes(); | 
|  | ASSERT_EQ(1u, changes.size()); | 
|  | EXPECT_EQ(syncer::SyncChange::ACTION_ADD, changes[0].change_type()); | 
|  | EXPECT_EQ(syncer::THEMES, changes[0].sync_data().GetDataType()); | 
|  |  | 
|  | const sync_pb::ThemeSpecifics& theme_specifics = | 
|  | changes[0].sync_data().GetSpecifics().theme(); | 
|  | EXPECT_FALSE(theme_specifics.use_custom_theme()); | 
|  | EXPECT_EQ(fake_theme_service_->GetAutogeneratedThemeColor(), | 
|  | theme_specifics.autogenerated_theme().color()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, UpdateThemeSpecifics_CurrentTheme_Policy) { | 
|  | // Set up theme service to use policy theme. | 
|  | fake_theme_service_->BuildAutogeneratedPolicyTheme(); | 
|  |  | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, syncer::SyncDataList(), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | // Applying policy theme doesn't trigger sync changes. | 
|  | const syncer::SyncChangeList& changes = fake_change_processor_->changes(); | 
|  | ASSERT_EQ(0u, changes.size()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, GetAllSyncDataForTesting_Extension) { | 
|  | // Set up theme service to use custom theme. | 
|  | fake_theme_service_->SetTheme(theme_extension_.get()); | 
|  |  | 
|  | syncer::SyncDataList data_list = | 
|  | theme_sync_service_->GetAllSyncDataForTesting(syncer::THEMES); | 
|  |  | 
|  | ASSERT_EQ(1u, data_list.size()); | 
|  | const sync_pb::ThemeSpecifics& theme_specifics = | 
|  | data_list[0].GetSpecifics().theme(); | 
|  | EXPECT_TRUE(theme_specifics.use_custom_theme()); | 
|  | EXPECT_EQ(theme_extension_->id(), theme_specifics.custom_theme_id()); | 
|  | EXPECT_EQ(theme_extension_->name(), theme_specifics.custom_theme_name()); | 
|  | EXPECT_EQ( | 
|  | extensions::ManifestURL::GetUpdateURL(theme_extension_.get()).spec(), | 
|  | theme_specifics.custom_theme_update_url()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, GetAllSyncDataForTesting_Autogenerated) { | 
|  | // Set up theme service to use autogenerated theme. | 
|  | fake_theme_service_->BuildAutogeneratedThemeFromColor( | 
|  | SkColorSetRGB(0, 0, 100)); | 
|  |  | 
|  | syncer::SyncDataList data_list = | 
|  | theme_sync_service_->GetAllSyncDataForTesting(syncer::THEMES); | 
|  |  | 
|  | ASSERT_EQ(1u, data_list.size()); | 
|  | const sync_pb::ThemeSpecifics& theme_specifics = | 
|  | data_list[0].GetSpecifics().theme(); | 
|  | EXPECT_FALSE(theme_specifics.use_custom_theme()); | 
|  | EXPECT_EQ(fake_theme_service_->GetAutogeneratedThemeColor(), | 
|  | theme_specifics.autogenerated_theme().color()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, GetAllSyncDataForTesting_Policy) { | 
|  | // Set up theme service to use policy theme. | 
|  | fake_theme_service_->BuildAutogeneratedPolicyTheme(); | 
|  |  | 
|  | // Themes applied through policy shouldn't be synced. | 
|  | syncer::SyncDataList data_list = | 
|  | theme_sync_service_->GetAllSyncDataForTesting(syncer::THEMES); | 
|  | ASSERT_EQ(0u, data_list.size()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, ProcessSyncThemeChange_Extension) { | 
|  | // Set up theme service to use default theme. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  | fake_theme_service_->MarkClean(); | 
|  |  | 
|  | // Start syncing. | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(sync_pb::ThemeSpecifics()), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | // Don't expect theme change initially because specifics are equal. | 
|  | EXPECT_FALSE(fake_theme_service_->is_dirty()); | 
|  |  | 
|  | // Change specifics to use custom theme and update. | 
|  | sync_pb::ThemeSpecifics theme_specifics; | 
|  | theme_specifics.set_use_custom_theme(true); | 
|  | theme_specifics.set_custom_theme_id(theme_extension_->id()); | 
|  | theme_specifics.set_custom_theme_name(kCustomThemeName); | 
|  | theme_specifics.set_custom_theme_update_url(kCustomThemeUrl); | 
|  | sync_pb::EntitySpecifics entity_specifics; | 
|  | entity_specifics.mutable_theme()->CopyFrom(theme_specifics); | 
|  | syncer::SyncChangeList change_list; | 
|  | change_list.push_back(syncer::SyncChange( | 
|  | FROM_HERE, syncer::SyncChange::ACTION_UPDATE, | 
|  | syncer::SyncData::CreateRemoteData( | 
|  | entity_specifics, syncer::ClientTagHash::FromHashed("unused")))); | 
|  | absl::optional<syncer::ModelError> process_error = | 
|  | theme_sync_service_->ProcessSyncChanges(FROM_HERE, change_list); | 
|  | EXPECT_FALSE(process_error.has_value()) << process_error.value().message(); | 
|  | EXPECT_EQ(fake_theme_service_->theme_extension(), theme_extension_.get()); | 
|  | // Don't show an infobar for theme installation. Regression test for | 
|  | // crbug.com/731688 | 
|  | EXPECT_FALSE(fake_theme_service_->might_show_infobar()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, ProcessSyncThemeChange_Autogenerated) { | 
|  | // Set up theme service to use default theme. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  | fake_theme_service_->MarkClean(); | 
|  |  | 
|  | // Start syncing. | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(sync_pb::ThemeSpecifics()), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | // Don't expect theme change initially because specifics are equal. | 
|  | EXPECT_FALSE(fake_theme_service_->is_dirty()); | 
|  |  | 
|  | // Change specifics to use custom theme and update. | 
|  | sync_pb::ThemeSpecifics theme_specifics; | 
|  | theme_specifics.set_use_custom_theme(false); | 
|  | theme_specifics.mutable_autogenerated_theme()->set_color( | 
|  | SkColorSetRGB(0, 0, 100)); | 
|  | sync_pb::EntitySpecifics entity_specifics; | 
|  | entity_specifics.mutable_theme()->CopyFrom(theme_specifics); | 
|  | syncer::SyncChangeList change_list; | 
|  | change_list.push_back(syncer::SyncChange( | 
|  | FROM_HERE, syncer::SyncChange::ACTION_UPDATE, | 
|  | syncer::SyncData::CreateRemoteData( | 
|  | entity_specifics, syncer::ClientTagHash::FromHashed("unused")))); | 
|  | absl::optional<syncer::ModelError> process_error = | 
|  | theme_sync_service_->ProcessSyncChanges(FROM_HERE, change_list); | 
|  | EXPECT_FALSE(process_error.has_value()) << process_error.value().message(); | 
|  | EXPECT_EQ(fake_theme_service_->GetAutogeneratedThemeColor(), | 
|  | SkColorSetRGB(0, 0, 100)); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, OnThemeChangeByUser_Extension) { | 
|  | // Set up theme service to use default theme. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  |  | 
|  | // Start syncing. | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(sync_pb::ThemeSpecifics()), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | const syncer::SyncChangeList& changes = fake_change_processor_->changes(); | 
|  | EXPECT_EQ(0u, changes.size()); | 
|  |  | 
|  | // Change current theme to custom theme and notify theme_sync_service_. | 
|  | fake_theme_service_->SetTheme(theme_extension_.get()); | 
|  | theme_sync_service_->OnThemeChanged(); | 
|  | EXPECT_EQ(1u, changes.size()); | 
|  | const sync_pb::ThemeSpecifics& change_specifics = | 
|  | changes[0].sync_data().GetSpecifics().theme(); | 
|  | EXPECT_TRUE(change_specifics.use_custom_theme()); | 
|  | EXPECT_EQ(theme_extension_->id(), change_specifics.custom_theme_id()); | 
|  | EXPECT_EQ(theme_extension_->name(), change_specifics.custom_theme_name()); | 
|  | EXPECT_EQ( | 
|  | extensions::ManifestURL::GetUpdateURL(theme_extension_.get()).spec(), | 
|  | change_specifics.custom_theme_update_url()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, OnThemeChangeByUser_Autogenerated) { | 
|  | // Set up theme service to use default theme. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  |  | 
|  | // Start syncing. | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(sync_pb::ThemeSpecifics()), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_FALSE(error.has_value()) << error.value().message(); | 
|  | const syncer::SyncChangeList& changes = fake_change_processor_->changes(); | 
|  | EXPECT_EQ(0u, changes.size()); | 
|  |  | 
|  | // Change current theme to custom theme and notify theme_sync_service_. | 
|  | fake_theme_service_->BuildAutogeneratedThemeFromColor( | 
|  | SkColorSetRGB(0, 0, 100)); | 
|  | theme_sync_service_->OnThemeChanged(); | 
|  | EXPECT_EQ(1u, changes.size()); | 
|  | const sync_pb::ThemeSpecifics& change_specifics = | 
|  | changes[0].sync_data().GetSpecifics().theme(); | 
|  | EXPECT_FALSE(change_specifics.use_custom_theme()); | 
|  | EXPECT_EQ(fake_theme_service_->GetAutogeneratedThemeColor(), | 
|  | SkColorSetRGB(0, 0, 100)); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, StopSync) { | 
|  | // Set up theme service to use default theme. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  |  | 
|  | // Start syncing. | 
|  | absl::optional<syncer::ModelError> merge_error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(sync_pb::ThemeSpecifics()), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_FALSE(merge_error.has_value()) << merge_error.value().message(); | 
|  | const syncer::SyncChangeList& changes = fake_change_processor_->changes(); | 
|  | EXPECT_EQ(0u, changes.size()); | 
|  |  | 
|  | // Stop syncing. | 
|  | theme_sync_service_->StopSyncing(syncer::THEMES); | 
|  |  | 
|  | // Change current theme to custom theme and notify theme_sync_service_. | 
|  | // No change is output because sync has stopped. | 
|  | fake_theme_service_->SetTheme(theme_extension_.get()); | 
|  | theme_sync_service_->OnThemeChanged(); | 
|  | EXPECT_EQ(0u, changes.size()); | 
|  |  | 
|  | // ProcessSyncChanges() should return error when sync has stopped. | 
|  | absl::optional<syncer::ModelError> process_error = | 
|  | theme_sync_service_->ProcessSyncChanges(FROM_HERE, changes); | 
|  | EXPECT_TRUE(process_error.has_value()); | 
|  | EXPECT_EQ("Theme syncable service is not started.", | 
|  | process_error.value().message()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, RestoreSystemThemeBitWhenChangeToCustomTheme) { | 
|  | // Initialize to use system theme. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  | sync_pb::ThemeSpecifics theme_specifics; | 
|  | theme_specifics.set_use_system_theme_by_default(true); | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(theme_specifics), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  |  | 
|  | // Change to custom theme and notify theme_sync_service_. | 
|  | // use_system_theme_by_default bit should be preserved. | 
|  | fake_theme_service_->SetTheme(theme_extension_.get()); | 
|  | theme_sync_service_->OnThemeChanged(); | 
|  | const syncer::SyncChangeList& changes = fake_change_processor_->changes(); | 
|  | EXPECT_EQ(1u, changes.size()); | 
|  | const sync_pb::ThemeSpecifics& change_specifics = | 
|  | changes[0].sync_data().GetSpecifics().theme(); | 
|  | EXPECT_TRUE(change_specifics.use_system_theme_by_default()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, DistinctSystemTheme) { | 
|  | fake_theme_service_->set_distinct_from_default_theme(true); | 
|  |  | 
|  | // Initialize to use native theme. | 
|  | fake_theme_service_->UseSystemTheme(); | 
|  | fake_theme_service_->MarkClean(); | 
|  | sync_pb::ThemeSpecifics theme_specifics; | 
|  | theme_specifics.set_use_system_theme_by_default(true); | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(theme_specifics), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_FALSE(fake_theme_service_->is_dirty()); | 
|  |  | 
|  | // Change to default theme and notify theme_sync_service_. | 
|  | // use_system_theme_by_default bit should be false. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  | theme_sync_service_->OnThemeChanged(); | 
|  | syncer::SyncChangeList& changes = fake_change_processor_->changes(); | 
|  | EXPECT_EQ(1u, changes.size()); | 
|  | EXPECT_FALSE(changes[0] | 
|  | .sync_data() | 
|  | .GetSpecifics() | 
|  | .theme() | 
|  | .use_system_theme_by_default()); | 
|  |  | 
|  | // Change to native theme and notify theme_sync_service_. | 
|  | // use_system_theme_by_default bit should be true. | 
|  | changes.clear(); | 
|  | fake_theme_service_->UseSystemTheme(); | 
|  | theme_sync_service_->OnThemeChanged(); | 
|  | EXPECT_EQ(1u, changes.size()); | 
|  | EXPECT_TRUE(changes[0] | 
|  | .sync_data() | 
|  | .GetSpecifics() | 
|  | .theme() | 
|  | .use_system_theme_by_default()); | 
|  | } | 
|  |  | 
|  | TEST_F(ThemeSyncableServiceTest, SystemThemeSameAsDefaultTheme) { | 
|  | fake_theme_service_->set_distinct_from_default_theme(false); | 
|  |  | 
|  | // Set up theme service to use default theme. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  |  | 
|  | // Initialize to use custom theme with use_system_theme_by_default set true. | 
|  | sync_pb::ThemeSpecifics theme_specifics; | 
|  | theme_specifics.set_use_custom_theme(true); | 
|  | theme_specifics.set_custom_theme_id(theme_extension_->id()); | 
|  | theme_specifics.set_custom_theme_name(kCustomThemeName); | 
|  | theme_specifics.set_custom_theme_update_url(kCustomThemeUrl); | 
|  | theme_specifics.set_use_system_theme_by_default(true); | 
|  | absl::optional<syncer::ModelError> error = | 
|  | theme_sync_service_->MergeDataAndStartSyncing( | 
|  | syncer::THEMES, MakeThemeDataList(theme_specifics), | 
|  | std::unique_ptr<syncer::SyncChangeProcessor>( | 
|  | new syncer::SyncChangeProcessorWrapperForTest( | 
|  | fake_change_processor_.get())), | 
|  | std::unique_ptr<syncer::SyncErrorFactory>( | 
|  | new syncer::SyncErrorFactoryMock())); | 
|  | EXPECT_EQ(fake_theme_service_->theme_extension(), theme_extension_.get()); | 
|  |  | 
|  | // Change to default theme and notify theme_sync_service_. | 
|  | // use_system_theme_by_default bit should be preserved. | 
|  | fake_theme_service_->UseDefaultTheme(); | 
|  | theme_sync_service_->OnThemeChanged(); | 
|  | const syncer::SyncChangeList& changes = fake_change_processor_->changes(); | 
|  | EXPECT_EQ(1u, changes.size()); | 
|  | const sync_pb::ThemeSpecifics& change_specifics = | 
|  | changes[0].sync_data().GetSpecifics().theme(); | 
|  | EXPECT_FALSE(change_specifics.use_custom_theme()); | 
|  | EXPECT_TRUE(change_specifics.use_system_theme_by_default()); | 
|  | } | 
|  |  | 
|  | TEST_F(PolicyInstalledThemeTest, InstallThemeByPolicy) { | 
|  | // Set up theme service to use custom theme that was installed by policy. | 
|  | fake_theme_service_->SetTheme(theme_extension_.get()); | 
|  |  | 
|  | syncer::SyncDataList data_list = | 
|  | theme_sync_service_->GetAllSyncDataForTesting(syncer::THEMES); | 
|  |  | 
|  | ASSERT_EQ(0u, data_list.size()); | 
|  | } |