| // Copyright 2023 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <string> |
| |
| #include "base/test/scoped_feature_list.h" |
| #include "chrome/browser/sync/test/integration/contact_info_helper.h" |
| #include "chrome/browser/sync/test/integration/fake_server_match_status_checker.h" |
| #include "chrome/browser/sync/test/integration/sync_service_impl_harness.h" |
| #include "chrome/browser/sync/test/integration/sync_test.h" |
| #include "components/autofill/core/browser/contact_info_sync_util.h" |
| #include "components/autofill/core/browser/personal_data_manager.h" |
| #include "components/autofill/core/common/autofill_features.h" |
| #include "components/sync/base/features.h" |
| #include "components/sync/base/model_type.h" |
| #include "components/sync/engine/loopback_server/persistent_unique_client_entity.h" |
| #include "components/sync/protocol/contact_info_specifics.pb.h" |
| #include "components/sync/protocol/entity_specifics.pb.h" |
| #include "components/sync/test/fake_server.h" |
| #include "content/public/test/browser_test.h" |
| #include "testing/gmock/include/gmock/gmock.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace { |
| |
| using autofill::AutofillProfile; |
| using contact_info_helper::BuildTestAccountProfile; |
| using contact_info_helper::PersonalDataManagerProfileChecker; |
| using testing::UnorderedElementsAre; |
| |
| // Helper class to wait until the fake server's ContactInfoSpecifics match a |
| // given predicate. |
| // Unfortunately, since protos don't have an equality operator, the comparisons |
| // are based on the `SerializeAsString()` representation of the specifics. |
| class FakeServerSpecificsChecker |
| : public fake_server::FakeServerMatchStatusChecker { |
| public: |
| using Matcher = testing::Matcher<std::vector<std::string>>; |
| |
| explicit FakeServerSpecificsChecker(const Matcher& matcher) |
| : matcher_(matcher) {} |
| |
| // StatusChangeChecker implementation. |
| bool IsExitConditionSatisfied(std::ostream* os) override { |
| std::vector<std::string> specifics; |
| for (const sync_pb::SyncEntity& entity : |
| fake_server()->GetSyncEntitiesByModelType(syncer::CONTACT_INFO)) { |
| specifics.push_back( |
| entity.specifics().contact_info().SerializeAsString()); |
| } |
| testing::StringMatchResultListener listener; |
| bool matches = testing::ExplainMatchResult(matcher_, specifics, &listener); |
| *os << listener.str(); |
| return matches; |
| } |
| |
| private: |
| const Matcher matcher_; |
| }; |
| |
| // Since the sync server operates in terms of entity specifics, this helper |
| // function converts a given `profile` to the equivalent ContactInfoSpecifics. |
| sync_pb::ContactInfoSpecifics AsContactInfoSpecifics( |
| const AutofillProfile& profile) { |
| return autofill::CreateContactInfoEntityDataFromAutofillProfile(profile) |
| ->specifics.contact_info(); |
| } |
| |
| // Adds the given `specifics` to the `fake_server` at creation time 0. |
| void AddSpecificsToServer(const sync_pb::ContactInfoSpecifics& specifics, |
| fake_server::FakeServer* fake_server) { |
| sync_pb::EntitySpecifics entity_specifics; |
| entity_specifics.mutable_contact_info()->CopyFrom(specifics); |
| fake_server->InjectEntity( |
| syncer::PersistentUniqueClientEntity::CreateFromSpecificsForTesting( |
| /*non_unique_name=*/"profile", /*client_tag=*/specifics.guid(), |
| /*entity_specifics=*/entity_specifics, /*creation_time=*/0, |
| /*last_modified_time=*/0)); |
| } |
| |
| class SingleClientContactInfoSyncTest : public SyncTest { |
| public: |
| SingleClientContactInfoSyncTest() : SyncTest(SINGLE_CLIENT) { |
| // The `PersonalDataManager` only loads `kAccount` profiles when |
| // kAutofillAccountProfilesUnionView is enabled. |
| features_.InitWithFeatures( |
| /*enabled_features=*/{syncer::kSyncEnableContactInfoDataType, |
| autofill::features:: |
| kAutofillAccountProfilesUnionView}, |
| /*disabled_features=*/{}); |
| } |
| |
| // In SINGLE_CLIENT tests, there's only a single PersonalDataManager. |
| autofill::PersonalDataManager* GetPersonalDataManager() const { |
| return contact_info_helper::GetPersonalDataManager(GetProfile(0)); |
| } |
| |
| private: |
| base::test::ScopedFeatureList features_; |
| }; |
| |
| IN_PROC_BROWSER_TEST_F(SingleClientContactInfoSyncTest, DownloadInitialData) { |
| const AutofillProfile kProfile = BuildTestAccountProfile(); |
| AddSpecificsToServer(AsContactInfoSpecifics(kProfile), GetFakeServer()); |
| ASSERT_TRUE(SetupSync()); |
| EXPECT_TRUE(PersonalDataManagerProfileChecker(GetPersonalDataManager(), |
| UnorderedElementsAre(kProfile)) |
| .Wait()); |
| } |
| |
| IN_PROC_BROWSER_TEST_F(SingleClientContactInfoSyncTest, UploadProfile) { |
| const AutofillProfile kProfile = BuildTestAccountProfile(); |
| ASSERT_TRUE(SetupSync()); |
| GetPersonalDataManager()->AddProfile(kProfile); |
| EXPECT_TRUE(FakeServerSpecificsChecker( |
| UnorderedElementsAre( |
| AsContactInfoSpecifics(kProfile).SerializeAsString())) |
| .Wait()); |
| } |
| |
| IN_PROC_BROWSER_TEST_F(SingleClientContactInfoSyncTest, ClearOnDisableSync) { |
| const AutofillProfile kProfile = BuildTestAccountProfile(); |
| AddSpecificsToServer(AsContactInfoSpecifics(kProfile), GetFakeServer()); |
| ASSERT_TRUE(SetupSync()); |
| ASSERT_TRUE(PersonalDataManagerProfileChecker(GetPersonalDataManager(), |
| UnorderedElementsAre(kProfile)) |
| .Wait()); |
| GetClient(0)->StopSyncServiceAndClearData(); |
| EXPECT_TRUE(PersonalDataManagerProfileChecker(GetPersonalDataManager(), |
| testing::IsEmpty()) |
| .Wait()); |
| } |
| |
| } // namespace |