OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2014 The Chromium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 #include "base/message_loop/message_loop.h" | |
6 #include "chrome/browser/sync/glue/device_info_sync_service.h" | |
7 #include "chrome/browser/sync/glue/local_device_info_provider_mock.h" | |
8 #include "content/public/test/test_browser_thread_bundle.h" | |
9 #include "sync/api/attachments/attachment_service_proxy_for_test.h" | |
10 #include "sync/api/sync_change.h" | |
11 #include "sync/api/sync_change_processor.h" | |
12 #include "sync/api/sync_change_processor_wrapper_for_test.h" | |
13 #include "sync/api/sync_error_factory_mock.h" | |
14 #include "testing/gtest/include/gtest/gtest.h" | |
15 | |
16 using syncer::AttachmentIdList; | |
17 using syncer::AttachmentServiceProxyForTest; | |
18 using syncer::ModelType; | |
19 using syncer::SyncChange; | |
20 using syncer::SyncChangeList; | |
21 using syncer::SyncChangeProcessor; | |
22 using syncer::SyncChangeProcessorWrapperForTest; | |
23 using syncer::SyncData; | |
24 using syncer::SyncDataList; | |
25 using syncer::SyncError; | |
26 using syncer::SyncErrorFactory; | |
27 using syncer::SyncErrorFactoryMock; | |
28 using syncer::SyncMergeResult; | |
29 | |
30 namespace browser_sync { | |
31 | |
32 namespace { | |
33 | |
34 class TestChangeProcessor : public SyncChangeProcessor { | |
35 public: | |
36 TestChangeProcessor() {} | |
37 virtual ~TestChangeProcessor() {} | |
38 | |
39 // SyncChangeProcessor implementation. | |
40 // Store a copy of all the changes passed in so we can examine them later. | |
41 virtual SyncError ProcessSyncChanges( | |
42 const tracked_objects::Location& from_here, | |
43 const SyncChangeList& change_list) OVERRIDE { | |
44 change_list_ = change_list; | |
45 return SyncError(); | |
46 } | |
47 | |
48 // This method isn't used in these tests. | |
49 virtual SyncDataList GetAllSyncData(ModelType type) const | |
50 OVERRIDE { | |
51 return SyncDataList(); | |
52 } | |
53 | |
54 size_t change_list_size() const { | |
55 return change_list_.size(); | |
56 } | |
57 | |
58 SyncChange::SyncChangeType change_type_at(size_t index) const { | |
59 return change_list_[index].change_type(); | |
60 } | |
61 | |
62 const sync_pb::DeviceInfoSpecifics& device_info_at(size_t index) const { | |
63 return change_list_[index].sync_data().GetSpecifics().device_info(); | |
64 } | |
65 | |
66 const std::string& cache_guid_at(size_t index) const { | |
67 return device_info_at(index).cache_guid(); | |
68 } | |
69 | |
70 const std::string& client_name_at(size_t index) const { | |
71 return device_info_at(index).client_name(); | |
72 } | |
73 | |
74 private: | |
75 SyncChangeList change_list_; | |
76 }; | |
77 | |
78 class DeviceInfoSyncServiceTest : public testing::Test { | |
79 public: | |
80 DeviceInfoSyncServiceTest() | |
81 : num_device_info_changed_callbacks_(0) {} | |
82 virtual ~DeviceInfoSyncServiceTest() {} | |
83 | |
84 virtual void SetUp() OVERRIDE { | |
85 local_device_.reset(new LocalDeviceInfoProviderMock( | |
86 "guid_1", | |
87 "client_1", | |
88 "Chromium 10k", | |
89 "Chrome 10k", | |
90 sync_pb::SyncEnums_DeviceType_TYPE_LINUX, | |
91 "device_id")); | |
92 sync_service_.reset(new DeviceInfoSyncService(local_device_.get())); | |
93 sync_processor_.reset(new TestChangeProcessor()); | |
94 // Register callbacks | |
95 notification_subscription_ = | |
96 sync_service_->RegisterOnDeviceInfoChangedCallback( | |
97 base::Bind(&DeviceInfoSyncServiceTest::OnDeviceInfoChanged, | |
98 base::Unretained(this))); | |
99 } | |
100 | |
101 virtual void TearDown() OVERRIDE { | |
102 } | |
103 | |
104 scoped_ptr<SyncChangeProcessor> PassProcessor() { | |
105 scoped_ptr<SyncChangeProcessorWrapperForTest> wrapper_( | |
106 new SyncChangeProcessorWrapperForTest(sync_processor_.get())); | |
107 return wrapper_.PassAs<SyncChangeProcessor>(); | |
pavely
2014/07/31 22:52:40
You can do:
return scoped_ptr<SyncChangeProcessor>
stanisc
2014/08/01 18:54:51
Done.
| |
108 } | |
109 | |
110 scoped_ptr<SyncErrorFactory> CreateAndPassSyncErrorFactory() { | |
111 return scoped_ptr<SyncErrorFactory>(new SyncErrorFactoryMock()); | |
112 } | |
113 | |
114 SyncData CreateRemoteData( | |
115 const std::string& client_id, const std::string& client_name) { | |
116 sync_pb::EntitySpecifics entity; | |
117 sync_pb::DeviceInfoSpecifics& specifics = *entity.mutable_device_info(); | |
118 | |
119 specifics.set_cache_guid(client_id); | |
120 specifics.set_client_name(client_name); | |
121 specifics.set_chrome_version("Chromium 10k"); | |
122 specifics.set_sync_user_agent("Chrome 10k"); | |
123 specifics.set_device_type(sync_pb::SyncEnums_DeviceType_TYPE_LINUX); | |
124 specifics.set_signin_scoped_device_id("device_id"); | |
125 | |
126 return SyncData::CreateRemoteData( | |
127 1, | |
128 entity, | |
129 base::Time(), | |
130 AttachmentIdList(), | |
131 AttachmentServiceProxyForTest::Create()); | |
132 } | |
133 | |
134 void AddInitialData(SyncDataList& sync_data_list, | |
135 const std::string& client_id, const std::string& client_name) { | |
136 SyncData sync_data = CreateRemoteData(client_id, client_name); | |
137 sync_data_list.push_back(sync_data); | |
138 } | |
139 | |
140 void AddChange(SyncChangeList& change_list, | |
141 SyncChange::SyncChangeType change_type, | |
142 const std::string& client_id, const std::string& client_name) { | |
143 SyncData sync_data = CreateRemoteData(client_id, client_name); | |
144 SyncChange sync_change(FROM_HERE, change_type, sync_data); | |
145 change_list.push_back(sync_change); | |
146 } | |
147 | |
148 protected: | |
149 void OnDeviceInfoChanged() { | |
150 num_device_info_changed_callbacks_++; | |
151 } | |
152 | |
153 int num_device_info_changed_callbacks_; | |
154 scoped_ptr<LocalDeviceInfoProviderMock> local_device_; | |
155 scoped_ptr<DeviceInfoSyncService> sync_service_; | |
156 scoped_ptr<TestChangeProcessor> sync_processor_; | |
157 scoped_ptr<DeviceInfoSyncService::Subscription> notification_subscription_; | |
158 content::TestBrowserThreadBundle thread_bundle_; | |
159 }; | |
160 | |
161 // Sync with empty initial data. | |
162 TEST_F(DeviceInfoSyncServiceTest, StartSyncEmptyInitialData) { | |
163 SyncMergeResult merge_result = sync_service_->MergeDataAndStartSyncing( | |
164 syncer::DEVICE_INFO, SyncDataList(), | |
165 PassProcessor(), CreateAndPassSyncErrorFactory()); | |
166 | |
167 EXPECT_EQ(0, merge_result.num_items_added()); | |
168 EXPECT_EQ(0, merge_result.num_items_modified()); | |
169 EXPECT_EQ(0, merge_result.num_items_deleted()); | |
170 EXPECT_EQ(1, merge_result.num_items_before_association()); | |
171 EXPECT_EQ(1, merge_result.num_items_after_association()); | |
172 EXPECT_EQ(SyncChange::ACTION_ADD, sync_processor_->change_type_at(0)); | |
173 | |
174 EXPECT_EQ(1U, sync_processor_->change_list_size()); | |
175 EXPECT_EQ("guid_1", sync_processor_->cache_guid_at(0)); | |
176 | |
177 // Should have one device info corresponding to local device info. | |
178 EXPECT_EQ(1U, sync_service_->GetAllSyncData(syncer::DEVICE_INFO).size()); | |
179 EXPECT_EQ(1U, sync_service_->GetAllDeviceInfo().size()); | |
180 EXPECT_TRUE(sync_service_->GetDeviceInfo("guid_1")); | |
181 EXPECT_FALSE(sync_service_->GetDeviceInfo("guid_0")); | |
182 } | |
183 | |
184 // Sync with initial data matching the local device data. | |
185 TEST_F(DeviceInfoSyncServiceTest, StartSyncMatchingInitialData) { | |
186 SyncDataList sync_data; | |
187 AddInitialData(sync_data, "guid_1", "client_1"); | |
188 | |
189 SyncMergeResult merge_result = sync_service_->MergeDataAndStartSyncing( | |
190 syncer::DEVICE_INFO, sync_data, | |
191 PassProcessor(), CreateAndPassSyncErrorFactory()); | |
192 EXPECT_EQ(0, merge_result.num_items_added()); | |
193 EXPECT_EQ(0, merge_result.num_items_modified()); | |
194 EXPECT_EQ(0, merge_result.num_items_deleted()); | |
195 EXPECT_EQ(1, merge_result.num_items_before_association()); | |
196 EXPECT_EQ(1, merge_result.num_items_after_association()); | |
197 | |
198 // No changes expected because the device info matches. | |
199 EXPECT_EQ(0U, sync_processor_->change_list_size()); | |
200 | |
201 EXPECT_EQ(1U, sync_service_->GetAllSyncData(syncer::DEVICE_INFO).size()); | |
202 EXPECT_EQ(1U, sync_service_->GetAllDeviceInfo().size()); | |
203 EXPECT_TRUE(sync_service_->GetDeviceInfo("guid_1")); | |
204 EXPECT_FALSE(sync_service_->GetDeviceInfo("guid_0")); | |
205 } | |
206 | |
207 // Sync with misc initial data. | |
208 TEST_F(DeviceInfoSyncServiceTest, StartSync) { | |
209 SyncDataList sync_data; | |
210 AddInitialData(sync_data, "guid_2", "foo"); | |
211 AddInitialData(sync_data, "guid_3", "bar"); | |
212 // This guid matches the local device but the client name is different. | |
213 AddInitialData(sync_data, "guid_1", "baz"); | |
214 | |
215 SyncMergeResult merge_result = sync_service_->MergeDataAndStartSyncing( | |
216 syncer::DEVICE_INFO, sync_data, | |
217 PassProcessor(), CreateAndPassSyncErrorFactory()); | |
218 | |
219 EXPECT_EQ(2, merge_result.num_items_added()); | |
220 EXPECT_EQ(1, merge_result.num_items_modified()); | |
221 EXPECT_EQ(0, merge_result.num_items_deleted()); | |
222 EXPECT_EQ(1, merge_result.num_items_before_association()); | |
223 EXPECT_EQ(3, merge_result.num_items_after_association()); | |
224 | |
225 EXPECT_EQ(1U, sync_processor_->change_list_size()); | |
226 EXPECT_EQ(SyncChange::ACTION_UPDATE, sync_processor_->change_type_at(0)); | |
227 EXPECT_EQ("client_1", sync_processor_->client_name_at(0)); | |
228 | |
229 EXPECT_EQ(3U, sync_service_->GetAllSyncData(syncer::DEVICE_INFO).size()); | |
230 EXPECT_EQ(3U, sync_service_->GetAllDeviceInfo().size()); | |
231 EXPECT_TRUE(sync_service_->GetDeviceInfo("guid_1")); | |
232 EXPECT_TRUE(sync_service_->GetDeviceInfo("guid_2")); | |
233 EXPECT_TRUE(sync_service_->GetDeviceInfo("guid_3")); | |
234 EXPECT_FALSE(sync_service_->GetDeviceInfo("guid_0")); | |
235 } | |
236 | |
237 // Process sync change with ACTION_ADD. | |
238 // Verify callback. | |
239 TEST_F(DeviceInfoSyncServiceTest, ProcessAddChange) { | |
240 EXPECT_EQ(0, num_device_info_changed_callbacks_); | |
241 | |
242 // Start with an empty initial data. | |
243 SyncMergeResult merge_result = sync_service_->MergeDataAndStartSyncing( | |
244 syncer::DEVICE_INFO, SyncDataList(), | |
245 PassProcessor(), CreateAndPassSyncErrorFactory()); | |
246 // There should be only one item corresponding to the local device | |
247 EXPECT_EQ(1, merge_result.num_items_after_association()); | |
248 EXPECT_EQ(0, num_device_info_changed_callbacks_); | |
249 | |
250 // Add a new device info with a non-matching guid. | |
251 SyncChangeList change_list; | |
252 AddChange(change_list, SyncChange::ACTION_ADD, "guid_2", "foo"); | |
253 | |
254 SyncError error = sync_service_->ProcessSyncChanges(FROM_HERE, change_list); | |
255 EXPECT_FALSE(error.IsSet()); | |
256 EXPECT_EQ(1, num_device_info_changed_callbacks_); | |
257 | |
258 EXPECT_EQ(2U, sync_service_->GetAllDeviceInfo().size()); | |
259 | |
260 EXPECT_TRUE(sync_service_->GetDeviceInfo("guid_1")); | |
261 EXPECT_TRUE(sync_service_->GetDeviceInfo("guid_2")); | |
262 EXPECT_FALSE(sync_service_->GetDeviceInfo("guid_0")); | |
263 } | |
264 | |
265 // Process multiple sync change with ACTION_UPDATE and ACTION_ADD. | |
266 // Verify that callback is called multiple times. | |
267 TEST_F(DeviceInfoSyncServiceTest, ProcessMultipleChanges) { | |
268 SyncDataList sync_data; | |
269 AddInitialData(sync_data, "guid_2", "foo"); | |
270 AddInitialData(sync_data, "guid_3", "bar"); | |
271 | |
272 SyncMergeResult merge_result = sync_service_->MergeDataAndStartSyncing( | |
273 syncer::DEVICE_INFO, sync_data, | |
274 PassProcessor(), CreateAndPassSyncErrorFactory()); | |
275 EXPECT_EQ(3, merge_result.num_items_after_association()); | |
276 | |
277 SyncChangeList change_list; | |
278 AddChange(change_list, SyncChange::ACTION_UPDATE, "guid_2", "foo_2"); | |
279 | |
280 SyncError error = sync_service_->ProcessSyncChanges(FROM_HERE, change_list); | |
281 EXPECT_FALSE(error.IsSet()); | |
282 | |
283 EXPECT_EQ(1, num_device_info_changed_callbacks_); | |
284 EXPECT_EQ(3U, sync_service_->GetAllDeviceInfo().size()); | |
285 EXPECT_EQ("foo_2", sync_service_->GetDeviceInfo("guid_2")->client_name()); | |
286 | |
287 change_list.clear(); | |
288 AddChange(change_list, SyncChange::ACTION_UPDATE, "guid_3", "bar_3"); | |
289 AddChange(change_list, SyncChange::ACTION_ADD, "guid_4", "baz_4"); | |
290 | |
291 error = sync_service_->ProcessSyncChanges(FROM_HERE, change_list); | |
292 EXPECT_FALSE(error.IsSet()); | |
293 | |
294 EXPECT_EQ(2, num_device_info_changed_callbacks_); | |
295 EXPECT_EQ(4U, sync_service_->GetAllDeviceInfo().size()); | |
296 EXPECT_EQ("bar_3", sync_service_->GetDeviceInfo("guid_3")->client_name()); | |
297 EXPECT_EQ("baz_4", sync_service_->GetDeviceInfo("guid_4")->client_name()); | |
298 } | |
299 | |
300 // Process update to the local device info and verify that it is ignored. | |
301 TEST_F(DeviceInfoSyncServiceTest, ProcessUpdateChangeMatchingLocalDevice) { | |
302 SyncMergeResult merge_result = sync_service_->MergeDataAndStartSyncing( | |
303 syncer::DEVICE_INFO, SyncDataList(), | |
304 PassProcessor(), CreateAndPassSyncErrorFactory()); | |
305 EXPECT_EQ(1, merge_result.num_items_after_association()); | |
306 EXPECT_EQ(0, num_device_info_changed_callbacks_); | |
307 | |
308 SyncChangeList change_list; | |
309 AddChange(change_list, SyncChange::ACTION_UPDATE, "guid_1", "foo_1"); | |
310 | |
311 SyncError error = sync_service_->ProcessSyncChanges(FROM_HERE, change_list); | |
312 EXPECT_FALSE(error.IsSet()); | |
313 // Callback shouldn't be sent in this case. | |
314 EXPECT_EQ(0, num_device_info_changed_callbacks_); | |
315 // Should still have the old local device Info. | |
316 EXPECT_EQ(1U, sync_service_->GetAllDeviceInfo().size()); | |
317 EXPECT_EQ("client_1", sync_service_->GetDeviceInfo("guid_1")->client_name()); | |
318 } | |
319 | |
320 // Process sync change with ACTION_DELETE. | |
321 TEST_F(DeviceInfoSyncServiceTest, ProcessDeleteChange) { | |
322 SyncDataList sync_data; | |
323 AddInitialData(sync_data, "guid_2", "foo"); | |
324 AddInitialData(sync_data, "guid_3", "bar"); | |
325 | |
326 SyncMergeResult merge_result = sync_service_->MergeDataAndStartSyncing( | |
327 syncer::DEVICE_INFO, sync_data, | |
328 PassProcessor(), CreateAndPassSyncErrorFactory()); | |
329 EXPECT_EQ(3, merge_result.num_items_after_association()); | |
330 | |
331 SyncChangeList change_list; | |
332 AddChange(change_list, SyncChange::ACTION_DELETE, "guid_2", "foo_2"); | |
333 | |
334 SyncError error = sync_service_->ProcessSyncChanges(FROM_HERE, change_list); | |
335 EXPECT_FALSE(error.IsSet()); | |
336 | |
337 EXPECT_EQ(2U, sync_service_->GetAllDeviceInfo().size()); | |
338 EXPECT_FALSE(sync_service_->GetDeviceInfo("guid_2")); | |
339 } | |
340 | |
341 // Process sync change with unexpected action. | |
342 TEST_F(DeviceInfoSyncServiceTest, ProcessInvalidChange) { | |
343 SyncMergeResult merge_result = sync_service_->MergeDataAndStartSyncing( | |
344 syncer::DEVICE_INFO, SyncDataList(), | |
345 PassProcessor(), CreateAndPassSyncErrorFactory()); | |
346 EXPECT_EQ(1, merge_result.num_items_after_association()); | |
347 EXPECT_EQ(0, num_device_info_changed_callbacks_); | |
348 | |
349 SyncChangeList change_list; | |
350 AddChange(change_list, (SyncChange::SyncChangeType)100, "guid_2", "foo_2"); | |
351 | |
352 SyncError error = sync_service_->ProcessSyncChanges(FROM_HERE, change_list); | |
353 EXPECT_TRUE(error.IsSet()); | |
354 | |
355 // The number of callback should still be zero. | |
356 EXPECT_EQ(0, num_device_info_changed_callbacks_); | |
357 EXPECT_EQ(1U, sync_service_->GetAllDeviceInfo().size()); | |
358 } | |
359 | |
360 // Process sync change after unsubscribing from notifications. | |
361 TEST_F(DeviceInfoSyncServiceTest, ProcessChangesAfterUnsubscribing) { | |
362 SyncMergeResult merge_result = sync_service_->MergeDataAndStartSyncing( | |
363 syncer::DEVICE_INFO, SyncDataList(), | |
364 PassProcessor(), CreateAndPassSyncErrorFactory()); | |
365 EXPECT_EQ(1, merge_result.num_items_after_association()); | |
366 EXPECT_EQ(0, num_device_info_changed_callbacks_); | |
367 | |
368 SyncChangeList change_list; | |
369 AddChange(change_list, SyncChange::ACTION_ADD, "guid_2", "foo_2"); | |
370 | |
371 // Unsubscribe before processing changes. | |
372 notification_subscription_.reset(); | |
373 | |
374 SyncError error = sync_service_->ProcessSyncChanges(FROM_HERE, change_list); | |
375 EXPECT_FALSE(error.IsSet()); | |
376 | |
377 // The number of callback should still be zero. | |
378 EXPECT_EQ(0, num_device_info_changed_callbacks_); | |
379 } | |
380 | |
381 } // namespace | |
382 | |
383 } // namespace browser_sync | |
OLD | NEW |