1 // Copyright (c) 2012 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.
5 #include "chrome/browser/sync/sync_prefs.h"
7 #include "base/command_line.h"
8 #include "base/logging.h"
9 #include "base/prefs/pref_member.h"
10 #include "base/prefs/pref_service.h"
11 #include "base/strings/string_number_conversions.h"
12 #include "base/values.h"
13 #include "build/build_config.h"
14 #include "chrome/browser/chrome_notification_types.h"
15 #include "chrome/browser/profiles/profile_io_data.h"
16 #include "chrome/browser/sync/profile_sync_service.h"
17 #include "chrome/common/chrome_switches.h"
18 #include "chrome/common/pref_names.h"
19 #include "components/user_prefs/pref_registry_syncable.h"
20 #include "content/public/browser/notification_details.h"
21 #include "content/public/browser/notification_source.h"
23 namespace browser_sync
{
25 SyncPrefObserver::~SyncPrefObserver() {}
27 SyncPrefs::SyncPrefs(PrefService
* pref_service
)
28 : pref_service_(pref_service
) {
31 // Watch the preference that indicates sync is managed so we can take
32 // appropriate action.
33 pref_sync_managed_
.Init(prefs::kSyncManaged
, pref_service_
,
34 base::Bind(&SyncPrefs::OnSyncManagedPrefChanged
,
35 base::Unretained(this)));
38 SyncPrefs::~SyncPrefs() {
39 DCHECK(CalledOnValidThread());
43 void SyncPrefs::RegisterProfilePrefs(
44 user_prefs::PrefRegistrySyncable
* registry
) {
45 registry
->RegisterBooleanPref(
46 prefs::kSyncHasSetupCompleted
,
48 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
49 registry
->RegisterBooleanPref(
50 prefs::kSyncSuppressStart
,
52 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
53 registry
->RegisterInt64Pref(
54 prefs::kSyncLastSyncedTime
,
56 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
58 // All datatypes are on by default, but this gets set explicitly
59 // when you configure sync (when turning it on), in
60 // ProfileSyncService::OnUserChoseDatatypes.
61 registry
->RegisterBooleanPref(
62 prefs::kSyncKeepEverythingSynced
,
64 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
66 syncer::ModelTypeSet user_types
= syncer::UserTypes();
68 // Include proxy types as well, as they can be individually selected,
69 // although they don't have sync representations.
70 user_types
.PutAll(syncer::ProxyTypes());
72 // Treat bookmarks specially.
73 RegisterDataTypePreferredPref(registry
, syncer::BOOKMARKS
, true);
74 user_types
.Remove(syncer::BOOKMARKS
);
76 // These two prefs are set from sync experiment to enable enhanced bookmarks.
77 registry
->RegisterBooleanPref(
78 prefs::kEnhancedBookmarksExperimentEnabled
,
80 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
82 registry
->RegisterStringPref(
83 prefs::kEnhancedBookmarksExtensionId
,
85 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
87 // All types are set to off by default, which forces a configuration to
88 // explicitly enable them. GetPreferredTypes() will ensure that any new
89 // implicit types are enabled when their pref group is, or via
90 // KeepEverythingSynced.
91 for (syncer::ModelTypeSet::Iterator it
= user_types
.First();
92 it
.Good(); it
.Inc()) {
93 RegisterDataTypePreferredPref(registry
, it
.Get(), false);
96 registry
->RegisterBooleanPref(
99 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
100 registry
->RegisterStringPref(
101 prefs::kSyncEncryptionBootstrapToken
,
103 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
104 registry
->RegisterStringPref(
105 prefs::kSyncKeystoreEncryptionBootstrapToken
,
107 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
108 #if defined(OS_CHROMEOS)
109 registry
->RegisterStringPref(
110 prefs::kSyncSpareBootstrapToken
,
112 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
115 registry
->RegisterBooleanPref(
116 prefs::kSyncHasAuthError
,
118 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
120 registry
->RegisterStringPref(
121 prefs::kSyncSessionsGUID
,
123 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
125 // We will start prompting people about new data types after the launch of
126 // SESSIONS - all previously launched data types are treated as if they are
127 // already acknowledged.
128 syncer::ModelTypeSet model_set
;
129 model_set
.Put(syncer::BOOKMARKS
);
130 model_set
.Put(syncer::PREFERENCES
);
131 model_set
.Put(syncer::PASSWORDS
);
132 model_set
.Put(syncer::AUTOFILL_PROFILE
);
133 model_set
.Put(syncer::AUTOFILL
);
134 model_set
.Put(syncer::THEMES
);
135 model_set
.Put(syncer::EXTENSIONS
);
136 model_set
.Put(syncer::NIGORI
);
137 model_set
.Put(syncer::SEARCH_ENGINES
);
138 model_set
.Put(syncer::APPS
);
139 model_set
.Put(syncer::APP_LIST
);
140 model_set
.Put(syncer::TYPED_URLS
);
141 model_set
.Put(syncer::SESSIONS
);
142 model_set
.Put(syncer::ARTICLES
);
143 registry
->RegisterListPref(prefs::kSyncAcknowledgedSyncTypes
,
144 syncer::ModelTypeSetToValue(model_set
),
145 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
148 void SyncPrefs::AddSyncPrefObserver(SyncPrefObserver
* sync_pref_observer
) {
149 DCHECK(CalledOnValidThread());
150 sync_pref_observers_
.AddObserver(sync_pref_observer
);
153 void SyncPrefs::RemoveSyncPrefObserver(SyncPrefObserver
* sync_pref_observer
) {
154 DCHECK(CalledOnValidThread());
155 sync_pref_observers_
.RemoveObserver(sync_pref_observer
);
158 void SyncPrefs::ClearPreferences() {
159 DCHECK(CalledOnValidThread());
160 pref_service_
->ClearPref(prefs::kSyncLastSyncedTime
);
161 pref_service_
->ClearPref(prefs::kSyncHasSetupCompleted
);
162 pref_service_
->ClearPref(prefs::kSyncEncryptionBootstrapToken
);
163 pref_service_
->ClearPref(prefs::kSyncKeystoreEncryptionBootstrapToken
);
165 // TODO(nick): The current behavior does not clear
166 // e.g. prefs::kSyncBookmarks. Is that really what we want?
169 bool SyncPrefs::HasSyncSetupCompleted() const {
170 DCHECK(CalledOnValidThread());
171 return pref_service_
->GetBoolean(prefs::kSyncHasSetupCompleted
);
174 void SyncPrefs::SetSyncSetupCompleted() {
175 DCHECK(CalledOnValidThread());
176 pref_service_
->SetBoolean(prefs::kSyncHasSetupCompleted
, true);
177 SetStartSuppressed(false);
180 bool SyncPrefs::SyncHasAuthError() const {
181 DCHECK(CalledOnValidThread());
182 return pref_service_
->GetBoolean(prefs::kSyncHasAuthError
);
185 void SyncPrefs::SetSyncAuthError(bool error
) {
186 DCHECK(CalledOnValidThread());
187 pref_service_
->SetBoolean(prefs::kSyncHasAuthError
, error
);
190 bool SyncPrefs::IsStartSuppressed() const {
191 DCHECK(CalledOnValidThread());
192 return pref_service_
->GetBoolean(prefs::kSyncSuppressStart
);
195 void SyncPrefs::SetStartSuppressed(bool is_suppressed
) {
196 DCHECK(CalledOnValidThread());
197 pref_service_
->SetBoolean(prefs::kSyncSuppressStart
, is_suppressed
);
200 std::string
SyncPrefs::GetGoogleServicesUsername() const {
201 DCHECK(CalledOnValidThread());
202 return pref_service_
->GetString(prefs::kGoogleServicesUsername
);
205 base::Time
SyncPrefs::GetLastSyncedTime() const {
206 DCHECK(CalledOnValidThread());
208 base::Time::FromInternalValue(
209 pref_service_
->GetInt64(prefs::kSyncLastSyncedTime
));
212 void SyncPrefs::SetLastSyncedTime(base::Time time
) {
213 DCHECK(CalledOnValidThread());
214 pref_service_
->SetInt64(prefs::kSyncLastSyncedTime
, time
.ToInternalValue());
217 bool SyncPrefs::HasKeepEverythingSynced() const {
218 DCHECK(CalledOnValidThread());
219 return pref_service_
->GetBoolean(prefs::kSyncKeepEverythingSynced
);
222 void SyncPrefs::SetKeepEverythingSynced(bool keep_everything_synced
) {
223 DCHECK(CalledOnValidThread());
224 pref_service_
->SetBoolean(prefs::kSyncKeepEverythingSynced
,
225 keep_everything_synced
);
228 syncer::ModelTypeSet
SyncPrefs::GetPreferredDataTypes(
229 syncer::ModelTypeSet registered_types
) const {
230 DCHECK(CalledOnValidThread());
232 // First remove any datatypes that are inconsistent with the current policies
233 // on the client (so that "keep everything synced" doesn't include them).
234 if (pref_service_
->HasPrefPath(prefs::kSavingBrowserHistoryDisabled
) &&
235 pref_service_
->GetBoolean(prefs::kSavingBrowserHistoryDisabled
)) {
236 registered_types
.Remove(syncer::TYPED_URLS
);
239 if (pref_service_
->GetBoolean(prefs::kSyncKeepEverythingSynced
)) {
240 return registered_types
;
243 syncer::ModelTypeSet preferred_types
;
244 for (syncer::ModelTypeSet::Iterator it
= registered_types
.First();
245 it
.Good(); it
.Inc()) {
246 if (GetDataTypePreferred(it
.Get())) {
247 preferred_types
.Put(it
.Get());
250 return ResolvePrefGroups(registered_types
, preferred_types
);
253 void SyncPrefs::SetPreferredDataTypes(
254 syncer::ModelTypeSet registered_types
,
255 syncer::ModelTypeSet preferred_types
) {
256 DCHECK(CalledOnValidThread());
257 DCHECK(registered_types
.HasAll(preferred_types
));
258 preferred_types
= ResolvePrefGroups(registered_types
, preferred_types
);
259 for (syncer::ModelTypeSet::Iterator i
= registered_types
.First();
261 SetDataTypePreferred(i
.Get(), preferred_types
.Has(i
.Get()));
265 bool SyncPrefs::IsManaged() const {
266 DCHECK(CalledOnValidThread());
267 return pref_service_
->GetBoolean(prefs::kSyncManaged
);
270 std::string
SyncPrefs::GetEncryptionBootstrapToken() const {
271 DCHECK(CalledOnValidThread());
272 return pref_service_
->GetString(prefs::kSyncEncryptionBootstrapToken
);
275 void SyncPrefs::SetEncryptionBootstrapToken(const std::string
& token
) {
276 DCHECK(CalledOnValidThread());
277 pref_service_
->SetString(prefs::kSyncEncryptionBootstrapToken
, token
);
280 std::string
SyncPrefs::GetKeystoreEncryptionBootstrapToken() const {
281 DCHECK(CalledOnValidThread());
282 return pref_service_
->GetString(
283 prefs::kSyncKeystoreEncryptionBootstrapToken
);
286 void SyncPrefs::SetKeystoreEncryptionBootstrapToken(const std::string
& token
) {
287 DCHECK(CalledOnValidThread());
288 pref_service_
->SetString(prefs::kSyncKeystoreEncryptionBootstrapToken
, token
);
291 std::string
SyncPrefs::GetSyncSessionsGUID() const {
292 DCHECK(CalledOnValidThread());
293 return pref_service_
->GetString(prefs::kSyncSessionsGUID
);
296 void SyncPrefs::SetSyncSessionsGUID(const std::string
& guid
) {
297 DCHECK(CalledOnValidThread());
298 pref_service_
->SetString(prefs::kSyncSessionsGUID
, guid
);
302 const char* SyncPrefs::GetPrefNameForDataType(syncer::ModelType data_type
) {
304 case syncer::BOOKMARKS
:
305 return prefs::kSyncBookmarks
;
306 case syncer::PASSWORDS
:
307 return prefs::kSyncPasswords
;
308 case syncer::PREFERENCES
:
309 return prefs::kSyncPreferences
;
310 case syncer::AUTOFILL
:
311 return prefs::kSyncAutofill
;
312 case syncer::AUTOFILL_PROFILE
:
313 return prefs::kSyncAutofillProfile
;
315 return prefs::kSyncThemes
;
316 case syncer::TYPED_URLS
:
317 return prefs::kSyncTypedUrls
;
318 case syncer::EXTENSION_SETTINGS
:
319 return prefs::kSyncExtensionSettings
;
320 case syncer::EXTENSIONS
:
321 return prefs::kSyncExtensions
;
322 case syncer::APP_LIST
:
323 return prefs::kSyncAppList
;
324 case syncer::APP_SETTINGS
:
325 return prefs::kSyncAppSettings
;
327 return prefs::kSyncApps
;
328 case syncer::SEARCH_ENGINES
:
329 return prefs::kSyncSearchEngines
;
330 case syncer::SESSIONS
:
331 return prefs::kSyncSessions
;
332 case syncer::APP_NOTIFICATIONS
:
333 return prefs::kSyncAppNotifications
;
334 case syncer::HISTORY_DELETE_DIRECTIVES
:
335 return prefs::kSyncHistoryDeleteDirectives
;
336 case syncer::SYNCED_NOTIFICATIONS
:
337 return prefs::kSyncSyncedNotifications
;
338 case syncer::SYNCED_NOTIFICATION_APP_INFO
:
339 return prefs::kSyncSyncedNotificationAppInfo
;
340 case syncer::DICTIONARY
:
341 return prefs::kSyncDictionary
;
342 case syncer::FAVICON_IMAGES
:
343 return prefs::kSyncFaviconImages
;
344 case syncer::FAVICON_TRACKING
:
345 return prefs::kSyncFaviconTracking
;
346 case syncer::MANAGED_USER_SETTINGS
:
347 return prefs::kSyncManagedUserSettings
;
348 case syncer::PROXY_TABS
:
349 return prefs::kSyncTabs
;
350 case syncer::PRIORITY_PREFERENCES
:
351 return prefs::kSyncPriorityPreferences
;
352 case syncer::MANAGED_USERS
:
353 return prefs::kSyncManagedUsers
;
354 case syncer::ARTICLES
:
355 return prefs::kSyncArticles
;
356 case syncer::MANAGED_USER_SHARED_SETTINGS
:
357 return prefs::kSyncManagedUserSharedSettings
;
365 #if defined(OS_CHROMEOS)
366 std::string
SyncPrefs::GetSpareBootstrapToken() const {
367 DCHECK(CalledOnValidThread());
368 return pref_service_
->GetString(prefs::kSyncSpareBootstrapToken
);
371 void SyncPrefs::SetSpareBootstrapToken(const std::string
& token
) {
372 DCHECK(CalledOnValidThread());
373 pref_service_
->SetString(prefs::kSyncSpareBootstrapToken
, token
);
377 void SyncPrefs::AcknowledgeSyncedTypes(syncer::ModelTypeSet types
) {
378 DCHECK(CalledOnValidThread());
379 // Add the types to the current set of acknowledged
380 // types, and then store the resulting set in prefs.
381 const syncer::ModelTypeSet acknowledged_types
=
383 syncer::ModelTypeSetFromValue(
384 *pref_service_
->GetList(prefs::kSyncAcknowledgedSyncTypes
)));
386 scoped_ptr
<base::ListValue
> value(
387 syncer::ModelTypeSetToValue(acknowledged_types
));
388 pref_service_
->Set(prefs::kSyncAcknowledgedSyncTypes
, *value
);
391 void SyncPrefs::OnSyncManagedPrefChanged() {
392 DCHECK(CalledOnValidThread());
393 FOR_EACH_OBSERVER(SyncPrefObserver
, sync_pref_observers_
,
394 OnSyncManagedPrefChange(*pref_sync_managed_
));
397 void SyncPrefs::SetManagedForTest(bool is_managed
) {
398 DCHECK(CalledOnValidThread());
399 pref_service_
->SetBoolean(prefs::kSyncManaged
, is_managed
);
402 syncer::ModelTypeSet
SyncPrefs::GetAcknowledgeSyncedTypesForTest() const {
403 DCHECK(CalledOnValidThread());
404 return syncer::ModelTypeSetFromValue(
405 *pref_service_
->GetList(prefs::kSyncAcknowledgedSyncTypes
));
408 void SyncPrefs::RegisterPrefGroups() {
409 pref_groups_
[syncer::APPS
].Put(syncer::APP_NOTIFICATIONS
);
410 pref_groups_
[syncer::APPS
].Put(syncer::APP_SETTINGS
);
411 pref_groups_
[syncer::APPS
].Put(syncer::APP_LIST
);
413 pref_groups_
[syncer::AUTOFILL
].Put(syncer::AUTOFILL_PROFILE
);
415 pref_groups_
[syncer::EXTENSIONS
].Put(syncer::EXTENSION_SETTINGS
);
417 pref_groups_
[syncer::PREFERENCES
].Put(syncer::DICTIONARY
);
418 pref_groups_
[syncer::PREFERENCES
].Put(syncer::PRIORITY_PREFERENCES
);
419 pref_groups_
[syncer::PREFERENCES
].Put(syncer::SEARCH_ENGINES
);
421 pref_groups_
[syncer::TYPED_URLS
].Put(syncer::HISTORY_DELETE_DIRECTIVES
);
422 if (!CommandLine::ForCurrentProcess()->HasSwitch(
423 switches::kHistoryDisableFullHistorySync
)) {
424 pref_groups_
[syncer::TYPED_URLS
].Put(syncer::SESSIONS
);
425 pref_groups_
[syncer::TYPED_URLS
].Put(syncer::FAVICON_IMAGES
);
426 pref_groups_
[syncer::TYPED_URLS
].Put(syncer::FAVICON_TRACKING
);
429 pref_groups_
[syncer::PROXY_TABS
].Put(syncer::SESSIONS
);
430 pref_groups_
[syncer::PROXY_TABS
].Put(syncer::FAVICON_IMAGES
);
431 pref_groups_
[syncer::PROXY_TABS
].Put(syncer::FAVICON_TRACKING
);
433 pref_groups_
[syncer::MANAGED_USER_SETTINGS
].Put(syncer::SESSIONS
);
435 // TODO(zea): put favicons in the bookmarks group as well once it handles
440 void SyncPrefs::RegisterDataTypePreferredPref(
441 user_prefs::PrefRegistrySyncable
* registry
,
442 syncer::ModelType type
,
444 const char* pref_name
= GetPrefNameForDataType(type
);
449 registry
->RegisterBooleanPref(
452 user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF
);
455 bool SyncPrefs::GetDataTypePreferred(syncer::ModelType type
) const {
456 DCHECK(CalledOnValidThread());
457 const char* pref_name
= GetPrefNameForDataType(type
);
462 if (type
== syncer::PROXY_TABS
&&
463 pref_service_
->GetUserPrefValue(pref_name
) == NULL
&&
464 pref_service_
->IsUserModifiablePreference(pref_name
)) {
465 // If there is no tab sync preference yet (i.e. newly enabled type),
466 // default to the session sync preference value.
467 pref_name
= GetPrefNameForDataType(syncer::SESSIONS
);
470 return pref_service_
->GetBoolean(pref_name
);
473 void SyncPrefs::SetDataTypePreferred(
474 syncer::ModelType type
, bool is_preferred
) {
475 DCHECK(CalledOnValidThread());
476 const char* pref_name
= GetPrefNameForDataType(type
);
481 pref_service_
->SetBoolean(pref_name
, is_preferred
);
484 syncer::ModelTypeSet
SyncPrefs::ResolvePrefGroups(
485 syncer::ModelTypeSet registered_types
,
486 syncer::ModelTypeSet types
) const {
487 DCHECK(registered_types
.HasAll(types
));
488 syncer::ModelTypeSet types_with_groups
= types
;
489 for (PrefGroupsMap::const_iterator i
= pref_groups_
.begin();
490 i
!= pref_groups_
.end(); ++i
) {
491 if (types
.Has(i
->first
))
492 types_with_groups
.PutAll(i
->second
);
494 types_with_groups
.RetainAll(registered_types
);
495 return types_with_groups
;
498 } // namespace browser_sync